summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBenjamin Otte <otte@redhat.com>2019-10-28 20:50:25 +0100
committerBenjamin Otte <otte@redhat.com>2019-12-12 07:47:44 +0100
commit6f92aa0de6373fbafa1723446b29923914799fed (patch)
tree049379efd73d9705f1197bd74cf5a20c010427cd
parentda06abeca124d0d302722c5fc02101e50de53531 (diff)
downloadgtk+-6f92aa0de6373fbafa1723446b29923914799fed.tar.gz
tests: Add testcolumnview
-rw-r--r--tests/meson.build1
-rw-r--r--tests/testcolumnview.c618
2 files changed, 619 insertions, 0 deletions
diff --git a/tests/meson.build b/tests/meson.build
index 17eada6284..2be5d96dae 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -26,6 +26,7 @@ gtk_tests = [
['testcolorchooser'],
['testcolorchooser2'],
['testcombo'],
+ ['testcolumnview'],
['testcombochange'],
['testcellrenderertext'],
['testdialog'],
diff --git a/tests/testcolumnview.c b/tests/testcolumnview.c
new file mode 100644
index 0000000000..f3cc3641d2
--- /dev/null
+++ b/tests/testcolumnview.c
@@ -0,0 +1,618 @@
+#include <gtk/gtk.h>
+
+GSList *pending = NULL;
+guint active = 0;
+
+static void
+loading_cb (GtkDirectoryList *dir,
+ GParamSpec *pspec,
+ gpointer unused)
+{
+ if (gtk_directory_list_is_loading (dir))
+ {
+ active++;
+ /* HACK: ensure loading finishes and the dir doesn't get destroyed */
+ g_object_ref (dir);
+ }
+ else
+ {
+ active--;
+ g_object_unref (dir);
+
+ while (active < 20 && pending)
+ {
+ GtkDirectoryList *dir2 = pending->data;
+ pending = g_slist_remove (pending, dir2);
+ gtk_directory_list_set_file (dir2, g_object_get_data (G_OBJECT (dir2), "file"));
+ g_object_unref (dir2);
+ }
+ }
+}
+
+static GtkDirectoryList *
+create_directory_list (GFile *file)
+{
+ GtkDirectoryList *dir;
+
+ dir = gtk_directory_list_new ("*",
+ NULL);
+ gtk_directory_list_set_io_priority (dir, G_PRIORITY_DEFAULT_IDLE);
+ g_signal_connect (dir, "notify::loading", G_CALLBACK (loading_cb), NULL);
+ g_assert (!gtk_directory_list_is_loading (dir));
+
+ if (active > 20)
+ {
+ g_object_set_data_full (G_OBJECT (dir), "file", g_object_ref (file), g_object_unref);
+ pending = g_slist_prepend (pending, g_object_ref (dir));
+ }
+ else
+ {
+ gtk_directory_list_set_file (dir, file);
+ }
+
+ return dir;
+}
+
+static char *
+get_file_path (GFileInfo *info)
+{
+ GFile *file;
+
+ file = G_FILE (g_file_info_get_attribute_object (info, "standard::file"));
+ return g_file_get_path (file);
+}
+
+static GListModel *
+create_list_model_for_directory (gpointer file)
+{
+ GtkSortListModel *sort;
+ GtkDirectoryList *dir;
+ GtkSorter *sorter;
+
+ if (g_file_query_file_type (file, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, NULL) != G_FILE_TYPE_DIRECTORY)
+ return NULL;
+
+ dir = create_directory_list (file);
+ sorter = gtk_string_sorter_new (gtk_cclosure_expression_new (G_TYPE_STRING, NULL, 0, NULL, (GCallback) get_file_path, NULL, NULL));
+ sort = gtk_sort_list_model_new (G_LIST_MODEL (dir), sorter);
+
+ g_object_unref (sorter);
+ g_object_unref (dir);
+ return G_LIST_MODEL (sort);
+}
+
+#if 0
+typedef struct _RowData RowData;
+struct _RowData
+{
+ GtkWidget *expander;
+ GtkWidget *icon;
+ GtkWidget *name;
+ GCancellable *cancellable;
+
+ GtkTreeListRow *current_item;
+};
+
+static void row_data_notify_item (GtkListItem *item,
+ GParamSpec *pspec,
+ RowData *data);
+static void
+row_data_unbind (RowData *data)
+{
+ if (data->current_item == NULL)
+ return;
+
+ if (data->cancellable)
+ {
+ g_cancellable_cancel (data->cancellable);
+ g_clear_object (&data->cancellable);
+ }
+
+ g_clear_object (&data->current_item);
+}
+
+static void
+row_data_update_info (RowData *data,
+ GFileInfo *info)
+{
+ GIcon *icon;
+ const char *thumbnail_path;
+
+ thumbnail_path = g_file_info_get_attribute_byte_string (info, G_FILE_ATTRIBUTE_THUMBNAIL_PATH);
+ if (thumbnail_path)
+ {
+ /* XXX: not async */
+ GFile *thumbnail_file = g_file_new_for_path (thumbnail_path);
+ icon = g_file_icon_new (thumbnail_file);
+ g_object_unref (thumbnail_file);
+ }
+ else
+ {
+ icon = g_file_info_get_icon (info);
+ }
+
+ gtk_widget_set_visible (data->icon, icon != NULL);
+ gtk_image_set_from_gicon (GTK_IMAGE (data->icon), icon);
+}
+
+static void
+copy_attribute (GFileInfo *to,
+ GFileInfo *from,
+ const gchar *attribute)
+{
+ GFileAttributeType type;
+ gpointer value;
+
+ if (g_file_info_get_attribute_data (from, attribute, &type, &value, NULL))
+ g_file_info_set_attribute (to, attribute, type, value);
+}
+
+static void
+row_data_got_thumbnail_info_cb (GObject *source,
+ GAsyncResult *res,
+ gpointer _data)
+{
+ RowData *data = _data; /* invalid if operation was cancelled */
+ GFile *file = G_FILE (source);
+ GFileInfo *queried, *info;
+
+ queried = g_file_query_info_finish (file, res, NULL);
+ if (queried == NULL)
+ return;
+
+ /* now we know row is valid */
+
+ info = gtk_tree_list_row_get_item (data->current_item);
+
+ copy_attribute (info, queried, G_FILE_ATTRIBUTE_THUMBNAIL_PATH);
+ copy_attribute (info, queried, G_FILE_ATTRIBUTE_THUMBNAILING_FAILED);
+ copy_attribute (info, queried, G_FILE_ATTRIBUTE_STANDARD_ICON);
+
+ g_object_unref (queried);
+
+ row_data_update_info (data, info);
+
+ g_clear_object (&data->cancellable);
+}
+
+static void
+row_data_bind (RowData *data,
+ GtkTreeListRow *item)
+{
+ GFileInfo *info;
+
+ row_data_unbind (data);
+
+ if (item == NULL)
+ return;
+
+ data->current_item = g_object_ref (item);
+
+ gtk_tree_expander_set_list_row (GTK_TREE_EXPANDER (data->expander), item);
+
+ info = gtk_tree_list_row_get_item (item);
+
+ if (!g_file_info_has_attribute (info, "filechooser::queried"))
+ {
+ data->cancellable = g_cancellable_new ();
+ g_file_info_set_attribute_boolean (info, "filechooser::queried", TRUE);
+ g_file_query_info_async (G_FILE (g_file_info_get_attribute_object (info, "standard::file")),
+ G_FILE_ATTRIBUTE_THUMBNAIL_PATH ","
+ G_FILE_ATTRIBUTE_THUMBNAILING_FAILED ","
+ G_FILE_ATTRIBUTE_STANDARD_ICON,
+ G_FILE_QUERY_INFO_NONE,
+ G_PRIORITY_DEFAULT,
+ data->cancellable,
+ row_data_got_thumbnail_info_cb,
+ data);
+ }
+
+ row_data_update_info (data, info);
+
+ gtk_label_set_label (GTK_LABEL (data->name), g_file_info_get_display_name (info));
+
+ g_object_unref (info);
+}
+
+static void
+row_data_notify_item (GtkListItem *item,
+ GParamSpec *pspec,
+ RowData *data)
+{
+ row_data_bind (data, gtk_list_item_get_item (item));
+}
+
+static void
+row_data_free (gpointer _data)
+{
+ RowData *data = _data;
+
+ row_data_unbind (data);
+
+ g_slice_free (RowData, data);
+}
+
+static void
+setup_widget (GtkListItem *list_item,
+ gpointer unused)
+{
+ GtkWidget *box, *child;
+ RowData *data;
+
+ data = g_slice_new0 (RowData);
+ g_signal_connect (list_item, "notify::item", G_CALLBACK (row_data_notify_item), data);
+ g_object_set_data_full (G_OBJECT (list_item), "row-data", data, row_data_free);
+
+ box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
+ gtk_container_add (GTK_CONTAINER (list_item), box);
+
+ child = gtk_label_new (NULL);
+ gtk_label_set_width_chars (GTK_LABEL (child), 5);
+ gtk_label_set_xalign (GTK_LABEL (child), 1.0);
+ g_object_bind_property (list_item, "position", child, "label", G_BINDING_SYNC_CREATE);
+ gtk_container_add (GTK_CONTAINER (box), child);
+
+ data->expander = gtk_tree_expander_new ();
+ gtk_container_add (GTK_CONTAINER (box), data->expander);
+
+ box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
+ gtk_tree_expander_set_child (GTK_TREE_EXPANDER (data->expander), box);
+
+ data->icon = gtk_image_new ();
+ gtk_container_add (GTK_CONTAINER (box), data->icon);
+
+ data->name = gtk_label_new (NULL);
+ gtk_label_set_max_width_chars (GTK_LABEL (data->name), 25);
+ gtk_label_set_ellipsize (GTK_LABEL (data->name), PANGO_ELLIPSIZE_END);
+ gtk_container_add (GTK_CONTAINER (box), data->name);
+}
+#endif
+
+static GListModel *
+create_list_model_for_file_info (gpointer file_info,
+ gpointer unused)
+{
+ GFile *file = G_FILE (g_file_info_get_attribute_object (file_info, "standard::file"));
+
+ if (file == NULL)
+ return NULL;
+
+ return create_list_model_for_directory (file);
+}
+
+static gboolean
+update_statusbar (GtkStatusbar *statusbar)
+{
+ GListModel *model = g_object_get_data (G_OBJECT (statusbar), "model");
+ GString *string = g_string_new (NULL);
+ guint n;
+ gboolean result = G_SOURCE_REMOVE;
+
+ gtk_statusbar_remove_all (statusbar, 0);
+
+ n = g_list_model_get_n_items (model);
+ g_string_append_printf (string, "%u", n);
+ if (GTK_IS_FILTER_LIST_MODEL (model))
+ {
+ guint n_unfiltered = g_list_model_get_n_items (gtk_filter_list_model_get_model (GTK_FILTER_LIST_MODEL (model)));
+ if (n != n_unfiltered)
+ g_string_append_printf (string, "/%u", n_unfiltered);
+ }
+ g_string_append (string, " items");
+
+ if (pending || active)
+ {
+ g_string_append_printf (string, " (%u directories remaining)", active + g_slist_length (pending));
+ result = G_SOURCE_CONTINUE;
+ }
+ result = G_SOURCE_CONTINUE;
+
+ gtk_statusbar_push (statusbar, 0, string->str);
+ g_free (string->str);
+
+ return result;
+}
+
+static gboolean
+match_file (gpointer item, gpointer data)
+{
+ GtkWidget *search_entry = data;
+ GFileInfo *info = gtk_tree_list_row_get_item (item);
+ GFile *file = G_FILE (g_file_info_get_attribute_object (info, "standard::file"));
+ char *path;
+ gboolean result;
+
+ path = g_file_get_path (file);
+
+ result = strstr (path, gtk_editable_get_text (GTK_EDITABLE (search_entry))) != NULL;
+
+ g_object_unref (info);
+ g_free (path);
+
+ return result;
+}
+
+const char *ui_file =
+"<?xml version='1.0' encoding='UTF-8'?>\n"
+"<interface>\n"
+" <object class='GtkColumnView' id='view'>\n"
+" <child>\n"
+" <object class='GtkColumnViewColumn'>\n"
+" <property name='title'>Name</property>\n"
+" <property name='factory'>\n"
+" <object class='GtkBuilderListItemFactory'>\n"
+" <property name='bytes'><![CDATA[\n"
+"<?xml version='1.0' encoding='UTF-8'?>\n"
+"<interface>\n"
+" <template class='GtkListItem'>\n"
+" <property name='child'>\n"
+" <object class='GtkTreeExpander' id='expander'>\n"
+" <binding name='list-row'>GtkListItem.item</binding>\n"
+" <property name='child'>\n"
+" <object class='GtkBox'>\n"
+" <child>\n"
+" <object class='GtkImage'>\n"
+" <binding name='gicon'>file-info(expander.item:GFileInfo, 'standard::icon', object):GIcon</binding>\n"
+" </object>\n"
+" </child>\n"
+" <child>\n"
+" <object class='GtkLabel'>\n"
+" <property name='halign'>start</property>\n"
+" <property name='label'>start</property>\n"
+" <binding name='label'>file-info(expander.item:GFileInfo, 'standard::display-name', string)</binding>\n"
+" </object>\n"
+" </child>\n"
+" </object>\n"
+" </property>\n"
+" </object>\n"
+" </property>\n"
+" </template>\n"
+"</interface>\n"
+" ]]></property>\n"
+" </object>\n"
+" </property>\n"
+" </object>\n"
+" </child>\n"
+" </object>\n"
+"</interface>\n";
+
+#define SIMPLE_STRING_FACTORY(attr, type) \
+"<?xml version='1.0' encoding='UTF-8'?>\n" \
+"<interface>\n" \
+" <template class='GtkListItem'>\n" \
+" <property name='child'>\n" \
+" <object class='GtkLabel'>\n" \
+" <property name='halign'>start</property>\n" \
+" <binding name='label'>file-info(GtkListItem.item:GtkTreeListRow.item:GFileInfo, '" attr "', " type")</binding>\n" \
+" </object>\n" \
+" </property>\n" \
+" </template>\n" \
+"</interface>\n" \
+
+#define BOOLEAN_FACTORY(attr) \
+"<?xml version='1.0' encoding='UTF-8'?>\n" \
+"<interface>\n" \
+" <template class='GtkListItem'>\n" \
+" <property name='child'>\n" \
+" <object class='GtkCheckButton'>\n" \
+" <binding name='active'>file-info(GtkListItem.item:GtkTreeListRow.item:GFileInfo, '" attr "', boolean)</binding>\n" \
+" </object>\n" \
+" </property>\n" \
+" </template>\n" \
+"</interface>\n" \
+
+#define ICON_FACTORY(attr) \
+"<?xml version='1.0' encoding='UTF-8'?>\n" \
+"<interface>\n" \
+" <template class='GtkListItem'>\n" \
+" <property name='child'>\n" \
+" <object class='GtkImage'>\n" \
+" <binding name='gicon'>file-info(GtkListItem.item:GtkTreeListRow.item:GFileInfo, '" attr "', object):GIcon</binding>\n" \
+" </object>\n" \
+" </property>\n" \
+" </template>\n" \
+"</interface>\n" \
+
+struct {
+ const char *title;
+ const char *factory_xml;
+} extra_columns[] = {
+ { "Type", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_TYPE, "uint32") },
+ { "Hidden", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_STANDARD_IS_HIDDEN) },
+ { "Backup", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_STANDARD_IS_BACKUP) },
+ { "Symlink", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_STANDARD_IS_SYMLINK) },
+ { "Virtual", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_STANDARD_IS_VIRTUAL) },
+ { "Volatile", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_STANDARD_IS_VOLATILE) },
+ { "Edit name", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_EDIT_NAME, "string") },
+ { "Copy name", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_COPY_NAME, "string") },
+ { "Description", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_DESCRIPTION, "string") },
+ { "Icon", ICON_FACTORY (G_FILE_ATTRIBUTE_STANDARD_ICON) },
+ { "Symbolic icon", ICON_FACTORY (G_FILE_ATTRIBUTE_STANDARD_SYMBOLIC_ICON) },
+ { "Content type", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, "string") },
+ { "Fast content type", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_FAST_CONTENT_TYPE, "string") },
+ { "Size", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_SIZE, "uint64") },
+ { "Allocated size", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_ALLOCATED_SIZE, "uint64") },
+ { "Target URI", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_TARGET_URI, "string") },
+ { "Sort order", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_STANDARD_SORT_ORDER, "int32") },
+ { "ETAG value", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_ETAG_VALUE, "string") },
+ { "File ID", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_ID_FILE, "string") },
+ { "Filesystem ID", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_ID_FILESYSTEM, "string") },
+ { "Read", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_READ) },
+ { "Write", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE) },
+ { "Execute", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_EXECUTE) },
+ { "Delete", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_DELETE) },
+ { "Trash", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_TRASH) },
+ { "Rename", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_ACCESS_CAN_RENAME) },
+ { "Can mount", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_MOUNTABLE_CAN_MOUNT) },
+ { "Can unmount", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_MOUNTABLE_CAN_UNMOUNT) },
+ { "Can eject", BOOLEAN_FACTORY (G_FILE_ATTRIBUTE_MOUNTABLE_CAN_EJECT) },
+ { "UNIX device", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_MOUNTABLE_UNIX_DEVICE, "uint32") },
+ { "UNIX device file", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_MOUNTABLE_UNIX_DEVICE_FILE, "string") },
+ { "owner", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_OWNER_USER, "string") },
+ { "owner (real)", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_OWNER_USER_REAL, "string") },
+ { "group", SIMPLE_STRING_FACTORY (G_FILE_ATTRIBUTE_OWNER_GROUP, "string") },
+ { "Preview icon", ICON_FACTORY (G_FILE_ATTRIBUTE_PREVIEW_ICON) },
+};
+
+#if 0
+#define G_FILE_ATTRIBUTE_MOUNTABLE_CAN_START "mountable::can-start" /* boolean */
+#define G_FILE_ATTRIBUTE_MOUNTABLE_CAN_START_DEGRADED "mountable::can-start-degraded" /* boolean */
+#define G_FILE_ATTRIBUTE_MOUNTABLE_CAN_STOP "mountable::can-stop" /* boolean */
+#define G_FILE_ATTRIBUTE_MOUNTABLE_START_STOP_TYPE "mountable::start-stop-type" /* uint32 (GDriveStartStopType) */
+#define G_FILE_ATTRIBUTE_MOUNTABLE_CAN_POLL "mountable::can-poll" /* boolean */
+#define G_FILE_ATTRIBUTE_MOUNTABLE_IS_MEDIA_CHECK_AUTOMATIC "mountable::is-media-check-automatic" /* boolean */
+#define G_FILE_ATTRIBUTE_TIME_MODIFIED "time::modified" /* uint64 */
+#define G_FILE_ATTRIBUTE_TIME_MODIFIED_USEC "time::modified-usec" /* uint32 */
+#define G_FILE_ATTRIBUTE_TIME_ACCESS "time::access" /* uint64 */
+#define G_FILE_ATTRIBUTE_TIME_ACCESS_USEC "time::access-usec" /* uint32 */
+#define G_FILE_ATTRIBUTE_TIME_CHANGED "time::changed" /* uint64 */
+#define G_FILE_ATTRIBUTE_TIME_CHANGED_USEC "time::changed-usec" /* uint32 */
+#define G_FILE_ATTRIBUTE_TIME_CREATED "time::created" /* uint64 */
+#define G_FILE_ATTRIBUTE_TIME_CREATED_USEC "time::created-usec" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_DEVICE "unix::device" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_INODE "unix::inode" /* uint64 */
+#define G_FILE_ATTRIBUTE_UNIX_MODE "unix::mode" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_NLINK "unix::nlink" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_UID "unix::uid" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_GID "unix::gid" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_RDEV "unix::rdev" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_BLOCK_SIZE "unix::block-size" /* uint32 */
+#define G_FILE_ATTRIBUTE_UNIX_BLOCKS "unix::blocks" /* uint64 */
+#define G_FILE_ATTRIBUTE_UNIX_IS_MOUNTPOINT "unix::is-mountpoint" /* boolean */
+#define G_FILE_ATTRIBUTE_DOS_IS_ARCHIVE "dos::is-archive" /* boolean */
+#define G_FILE_ATTRIBUTE_DOS_IS_SYSTEM "dos::is-system" /* boolean */
+#define G_FILE_ATTRIBUTE_DOS_IS_MOUNTPOINT "dos::is-mountpoint" /* boolean */
+#define G_FILE_ATTRIBUTE_DOS_REPARSE_POINT_TAG "dos::reparse-point-tag" /* uint32 */
+#define G_FILE_ATTRIBUTE_THUMBNAIL_PATH "thumbnail::path" /* bytestring */
+#define G_FILE_ATTRIBUTE_THUMBNAILING_FAILED "thumbnail::failed" /* boolean */
+#define G_FILE_ATTRIBUTE_THUMBNAIL_IS_VALID "thumbnail::is-valid" /* boolean */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_SIZE "filesystem::size" /* uint64 */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_FREE "filesystem::free" /* uint64 */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_USED "filesystem::used" /* uint64 */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_TYPE "filesystem::type" /* string */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_READONLY "filesystem::readonly" /* boolean */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_USE_PREVIEW "filesystem::use-preview" /* uint32 (GFilesystemPreviewType) */
+#define G_FILE_ATTRIBUTE_FILESYSTEM_REMOTE "filesystem::remote" /* boolean */
+#define G_FILE_ATTRIBUTE_GVFS_BACKEND "gvfs::backend" /* string */
+#define G_FILE_ATTRIBUTE_SELINUX_CONTEXT "selinux::context" /* string */
+#define G_FILE_ATTRIBUTE_TRASH_ITEM_COUNT "trash::item-count" /* uint32 */
+#define G_FILE_ATTRIBUTE_TRASH_ORIG_PATH "trash::orig-path" /* byte string */
+#define G_FILE_ATTRIBUTE_TRASH_DELETION_DATE "trash::deletion-date" /* string */
+#define G_FILE_ATTRIBUTE_RECENT_MODIFIED "recent::modified" /* int64 (time_t) */
+#endif
+
+const char *factory_ui =
+"<?xml version='1.0' encoding='UTF-8'?>\n" \
+"<interface>\n" \
+" <template class='GtkListItem'>\n" \
+" <property name='child'>\n" \
+" <object class='GtkLabel'>\n" \
+" <binding name='label'>GtkListItem.item:GtkColumnViewColumn.title</binding>\n" \
+" </object>\n" \
+" </property>\n" \
+" </template>\n" \
+"</interface>\n";
+
+static void
+add_extra_columns (GtkColumnView *view)
+{
+ GtkColumnViewColumn *column;
+ GBytes *bytes;
+ guint i;
+
+ for (i = 0; i < G_N_ELEMENTS(extra_columns); i++)
+ {
+ bytes = g_bytes_new_static (extra_columns[i].factory_xml, strlen (extra_columns[i].factory_xml));
+ column = gtk_column_view_column_new_with_factory (extra_columns[i].title,
+ gtk_builder_list_item_factory_new_from_bytes (NULL, bytes));
+ g_bytes_unref (bytes);
+ gtk_column_view_append_column (view, column);
+ }
+}
+
+static void
+search_changed_cb (GtkSearchEntry *entry,
+ GtkFilter *custom_filter)
+{
+ gtk_filter_changed (custom_filter, GTK_FILTER_CHANGE_DIFFERENT);
+}
+
+int
+main (int argc, char *argv[])
+{
+ GtkWidget *win, *hbox, *vbox, *sw, *view, *list, *search_entry, *statusbar;
+ GListModel *dirmodel;
+ GtkTreeListModel *tree;
+ GtkFilterListModel *filter;
+ GtkFilter *custom_filter;
+ GFile *root;
+ GtkBuilder *builder;
+
+ gtk_init ();
+
+ win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
+ gtk_window_set_default_size (GTK_WINDOW (win), 800, 600);
+ g_signal_connect (win, "destroy", G_CALLBACK (gtk_main_quit), win);
+
+ hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
+ gtk_container_add (GTK_CONTAINER (win), hbox);
+
+ vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+ gtk_container_add (GTK_CONTAINER (hbox), vbox);
+
+ search_entry = gtk_search_entry_new ();
+ gtk_container_add (GTK_CONTAINER (vbox), search_entry);
+
+ sw = gtk_scrolled_window_new (NULL, NULL);
+ gtk_widget_set_hexpand (sw, TRUE);
+ gtk_widget_set_vexpand (sw, TRUE);
+ gtk_search_entry_set_key_capture_widget (GTK_SEARCH_ENTRY (search_entry), sw);
+ gtk_container_add (GTK_CONTAINER (vbox), sw);
+
+ builder = gtk_builder_new_from_string (ui_file, -1);
+ view = GTK_WIDGET (gtk_builder_get_object (builder, "view"));
+ add_extra_columns (GTK_COLUMN_VIEW (view));
+ gtk_container_add (GTK_CONTAINER (sw), view);
+ g_object_unref (builder);
+
+ if (argc > 1)
+ root = g_file_new_for_commandline_arg (argv[1]);
+ else
+ root = g_file_new_for_path (g_get_current_dir ());
+ dirmodel = create_list_model_for_directory (root);
+ tree = gtk_tree_list_model_new (FALSE,
+ dirmodel,
+ TRUE,
+ create_list_model_for_file_info,
+ NULL, NULL);
+ g_object_unref (dirmodel);
+ g_object_unref (root);
+
+ custom_filter = gtk_custom_filter_new (match_file, search_entry, NULL);
+ filter = gtk_filter_list_model_new (G_LIST_MODEL (tree), custom_filter);
+ g_signal_connect (search_entry, "search-changed", G_CALLBACK (search_changed_cb), custom_filter);
+ g_object_unref (custom_filter);
+
+ gtk_column_view_set_model (GTK_COLUMN_VIEW (view), G_LIST_MODEL (filter));
+
+ statusbar = gtk_statusbar_new ();
+ gtk_widget_add_tick_callback (statusbar, (GtkTickCallback) update_statusbar, NULL, NULL);
+ g_object_set_data (G_OBJECT (statusbar), "model", filter);
+ g_signal_connect_swapped (filter, "items-changed", G_CALLBACK (update_statusbar), statusbar);
+ update_statusbar (GTK_STATUSBAR (statusbar));
+ gtk_container_add (GTK_CONTAINER (vbox), statusbar);
+
+ g_object_unref (tree);
+ g_object_unref (filter);
+
+ list = gtk_list_view_new_with_factory (
+ gtk_builder_list_item_factory_new_from_bytes (NULL, g_bytes_new_static (factory_ui, strlen (factory_ui))));
+ gtk_list_view_set_model (GTK_LIST_VIEW (list), gtk_column_view_get_columns (GTK_COLUMN_VIEW (view)));
+ gtk_container_add (GTK_CONTAINER (hbox), list);
+
+ gtk_widget_show (win);
+
+ gtk_main ();
+
+ return 0;
+}