[gnome-builder] tree: add cell renderer for vcs file status
- From: Christian Hergert <chergert src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-builder] tree: add cell renderer for vcs file status
- Date: Fri, 28 Jun 2019 19:43:04 +0000 (UTC)
commit 6862853f7831b342785d4db1aa40c5f5b1d68411
Author: Christian Hergert <chergert redhat com>
Date: Fri Jun 28 12:41:12 2019 -0700
tree: add cell renderer for vcs file status
src/libide/tree/ide-cell-renderer-status.c | 213 +++++++++++++++++++++++++++++
src/libide/tree/ide-cell-renderer-status.h | 37 +++++
src/libide/tree/ide-tree-node.c | 24 ++++
src/libide/tree/ide-tree-node.h | 14 ++
src/libide/tree/ide-tree.c | 26 ++++
src/libide/tree/meson.build | 6 +-
6 files changed, 319 insertions(+), 1 deletion(-)
---
diff --git a/src/libide/tree/ide-cell-renderer-status.c b/src/libide/tree/ide-cell-renderer-status.c
new file mode 100644
index 000000000..6c3781e34
--- /dev/null
+++ b/src/libide/tree/ide-cell-renderer-status.c
@@ -0,0 +1,213 @@
+/* ide-cell-renderer-status.c
+ *
+ * Copyright 2019 Christian Hergert <chergert redhat 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 3 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/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#define G_LOG_DOMAIN "ide-cell-renderer-status"
+
+#include "config.h"
+
+#include <math.h>
+
+#include "ide-cell-renderer-status.h"
+
+#define CELL_HEIGHT 16
+#define CELL_WIDTH 16
+
+struct _IdeCellRendererStatus
+{
+ GtkCellRenderer parent_instance;
+ IdeTreeNodeFlags flags;
+};
+
+enum {
+ PROP_0,
+ PROP_FLAGS,
+ N_PROPS
+};
+
+G_DEFINE_TYPE (IdeCellRendererStatus, ide_cell_renderer_status, GTK_TYPE_CELL_RENDERER)
+
+static GParamSpec *properties [N_PROPS];
+
+static void
+ide_cell_renderer_status_get_preferred_height (GtkCellRenderer *cell,
+ GtkWidget *widget,
+ gint *min_size,
+ gint *nat_size)
+{
+ g_assert (IDE_IS_CELL_RENDERER_STATUS (cell));
+ g_assert (GTK_IS_WIDGET (widget));
+
+ if (min_size)
+ *min_size = CELL_HEIGHT;
+
+ if (nat_size)
+ *nat_size = CELL_HEIGHT;
+}
+
+static void
+ide_cell_renderer_status_get_preferred_width (GtkCellRenderer *cell,
+ GtkWidget *widget,
+ gint *min_size,
+ gint *nat_size)
+{
+ gint xpad;
+
+ g_assert (IDE_IS_CELL_RENDERER_STATUS (cell));
+ g_assert (GTK_IS_WIDGET (widget));
+
+ g_object_get (cell, "xpad", &xpad, NULL);
+
+ if (min_size)
+ *min_size = CELL_WIDTH + xpad;
+
+ if (nat_size)
+ *nat_size = CELL_WIDTH + xpad;
+}
+
+static void
+ide_cell_renderer_status_render (GtkCellRenderer *cell,
+ cairo_t *cr,
+ GtkWidget *widget,
+ const GdkRectangle *bg_area,
+ const GdkRectangle *cell_area,
+ GtkCellRendererState state)
+{
+ IdeCellRendererStatus *self = (IdeCellRendererStatus *)cell;
+ GtkStyleContext *style_context;
+ GdkRGBA color;
+ gint xpad;
+
+ g_assert (IDE_IS_CELL_RENDERER_STATUS (self));
+ g_assert (cr != NULL);
+ g_assert (GTK_IS_WIDGET (widget));
+ g_assert (bg_area != NULL);
+ g_assert (cell_area != NULL);
+
+ if (self->flags == 0)
+ return;
+
+ style_context = gtk_widget_get_style_context (widget);
+ gtk_style_context_save (style_context);
+
+ if (state & GTK_CELL_RENDERER_SELECTED)
+ gtk_style_context_set_state (style_context,
+ gtk_style_context_get_state (style_context) & GTK_STATE_FLAG_SELECTED);
+ gtk_style_context_get_color (style_context,
+ gtk_style_context_get_state (style_context),
+ &color);
+ gdk_cairo_set_source_rgba (cr, &color);
+
+ g_object_get (cell, "xpad", &xpad, NULL);
+
+ cairo_arc (cr,
+ cell_area->x + cell_area->width - (CELL_WIDTH/2) - xpad,
+ cell_area->y + (cell_area->height / 2),
+ 3,
+ 0,
+ M_PI * 2);
+
+ if (self->flags & IDE_TREE_NODE_FLAGS_ADDED)
+ cairo_fill_preserve (cr);
+
+ cairo_stroke (cr);
+
+ gtk_style_context_restore (style_context);
+}
+
+static void
+ide_cell_renderer_status_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ IdeCellRendererStatus *self = IDE_CELL_RENDERER_STATUS (object);
+
+ switch (prop_id)
+ {
+ case PROP_FLAGS:
+ g_value_set_uint (value, self->flags);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ide_cell_renderer_status_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ IdeCellRendererStatus *self = IDE_CELL_RENDERER_STATUS (object);
+
+ switch (prop_id)
+ {
+ case PROP_FLAGS:
+ self->flags = g_value_get_uint (value);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+ide_cell_renderer_status_class_init (IdeCellRendererStatusClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkCellRendererClass *renderer_class = GTK_CELL_RENDERER_CLASS (klass);
+
+ object_class->get_property = ide_cell_renderer_status_get_property;
+ object_class->set_property = ide_cell_renderer_status_set_property;
+
+ renderer_class->get_preferred_height = ide_cell_renderer_status_get_preferred_height;
+ renderer_class->get_preferred_width = ide_cell_renderer_status_get_preferred_width;
+ renderer_class->render = ide_cell_renderer_status_render;
+
+ properties [PROP_FLAGS] =
+ g_param_spec_uint ("flags",
+ "Flags",
+ "The flags for the state",
+ 0, G_MAXUINT, 0,
+ (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+
+ g_object_class_install_properties (object_class, N_PROPS, properties);
+}
+
+static void
+ide_cell_renderer_status_init (IdeCellRendererStatus *self)
+{
+}
+
+GtkCellRenderer *
+ide_cell_renderer_status_new (void)
+{
+ return g_object_new (IDE_TYPE_CELL_RENDERER_STATUS, NULL);
+}
+
+void
+ide_cell_renderer_status_set_flags (IdeCellRendererStatus *self,
+ IdeTreeNodeFlags flags)
+{
+ g_return_if_fail (IDE_IS_CELL_RENDERER_STATUS (self));
+
+ self->flags = flags;
+}
diff --git a/src/libide/tree/ide-cell-renderer-status.h b/src/libide/tree/ide-cell-renderer-status.h
new file mode 100644
index 000000000..1ed489c58
--- /dev/null
+++ b/src/libide/tree/ide-cell-renderer-status.h
@@ -0,0 +1,37 @@
+/* ide-cell-renderer-status.h
+ *
+ * Copyright 2019 Christian Hergert <chergert redhat 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 3 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/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#pragma once
+
+#include <gtk/gtk.h>
+
+#include "ide-tree-node.h"
+
+G_BEGIN_DECLS
+
+#define IDE_TYPE_CELL_RENDERER_STATUS (ide_cell_renderer_status_get_type())
+
+G_DECLARE_FINAL_TYPE (IdeCellRendererStatus, ide_cell_renderer_status, IDE, CELL_RENDERER_STATUS,
GtkCellRenderer)
+
+GtkCellRenderer *ide_cell_renderer_status_new (void);
+void ide_cell_renderer_status_set_flags (IdeCellRendererStatus *self,
+ IdeTreeNodeFlags flags);
+
+G_END_DECLS
diff --git a/src/libide/tree/ide-tree-node.c b/src/libide/tree/ide-tree-node.c
index 5d823605d..12499c6c4 100644
--- a/src/libide/tree/ide-tree-node.c
+++ b/src/libide/tree/ide-tree-node.c
@@ -81,6 +81,9 @@ struct _IdeTreeNode
GdkRGBA background;
GdkRGBA foreground;
+ /* Flags for state cell renderer */
+ IdeTreeNodeFlags flags;
+
/* When did we start loading? This is used to avoid drawing "Loading..."
* when the tree loads really quickly. Otherwise, we risk looking janky
* when the loads are quite fast.
@@ -1995,3 +1998,24 @@ ide_tree_node_set_use_markup (IdeTreeNode *self,
g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_USE_MARKUP]);
}
}
+
+IdeTreeNodeFlags
+ide_tree_node_get_flags (IdeTreeNode *self)
+{
+ g_return_val_if_fail (IDE_IS_TREE_NODE (self), 0);
+
+ return self->flags;
+}
+
+void
+ide_tree_node_set_flags (IdeTreeNode *self,
+ IdeTreeNodeFlags flags)
+{
+ g_return_if_fail (IDE_IS_TREE_NODE (self));
+
+ if (self->flags != flags)
+ {
+ self->flags = flags;
+ ide_tree_node_emit_changed (self);
+ }
+}
diff --git a/src/libide/tree/ide-tree-node.h b/src/libide/tree/ide-tree-node.h
index 0aa301b11..eed24d0cf 100644
--- a/src/libide/tree/ide-tree-node.h
+++ b/src/libide/tree/ide-tree-node.h
@@ -36,6 +36,15 @@ typedef enum
IDE_TREE_NODE_VISIT_CHILDREN = 0x3,
} IdeTreeNodeVisit;
+typedef enum
+{
+ IDE_TREE_NODE_FLAGS_NONE = 0,
+ IDE_TREE_NODE_FLAGS_DESCENDANT = 1 << 0,
+ IDE_TREE_NODE_FLAGS_ADDED = 1 << 1,
+ IDE_TREE_NODE_FLAGS_CHANGED = 1 << 2,
+ IDE_TREE_NODE_FLAGS_REMOVED = 1 << 3,
+} IdeTreeNodeFlags;
+
/**
* IdeTreeTraverseFunc:
* @node: an #IdeTreeNode
@@ -198,5 +207,10 @@ gboolean ide_tree_node_get_use_markup (IdeTreeNode *self);
IDE_AVAILABLE_IN_3_32
void ide_tree_node_set_use_markup (IdeTreeNode *self,
gboolean use_markup);
+IDE_AVAILABLE_IN_3_34
+void ide_tree_node_set_flags (IdeTreeNode *self,
+ IdeTreeNodeFlags flags);
+IDE_AVAILABLE_IN_3_34
+IdeTreeNodeFlags ide_tree_node_get_flags (IdeTreeNode *self);
G_END_DECLS
diff --git a/src/libide/tree/ide-tree.c b/src/libide/tree/ide-tree.c
index 03323e254..0a4838018 100644
--- a/src/libide/tree/ide-tree.c
+++ b/src/libide/tree/ide-tree.c
@@ -26,6 +26,7 @@
#include <libide-core.h>
#include <libide-threading.h>
+#include "ide-cell-renderer-status.h"
#include "ide-tree.h"
#include "ide-tree-model.h"
#include "ide-tree-node.h"
@@ -118,6 +119,25 @@ ide_tree_select (IdeTree *self,
gtk_tree_selection_select_path (selection, path);
}
+static void
+state_cell_func (GtkCellLayout *layout,
+ GtkCellRenderer *cell,
+ GtkTreeModel *model,
+ GtkTreeIter *iter,
+ gpointer user_data)
+{
+ IdeTreeNodeFlags flags = 0;
+ IdeTreeNode *node;
+
+ g_assert (IDE_IS_TREE (user_data));
+ g_assert (IDE_IS_TREE_MODEL (model));
+
+ if ((node = ide_tree_model_get_node (IDE_TREE_MODEL (model), iter)))
+ flags = ide_tree_node_get_flags (node);
+
+ ide_cell_renderer_status_set_flags (IDE_CELL_RENDERER_STATUS (cell), flags);
+}
+
static void
text_cell_func (GtkCellLayout *layout,
GtkCellRenderer *cell,
@@ -565,6 +585,12 @@ ide_tree_init (IdeTree *self)
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (column), cell, TRUE);
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (column), cell, text_cell_func, self, NULL);
+ cell = g_object_new (IDE_TYPE_CELL_RENDERER_STATUS,
+ "xpad", 3,
+ NULL);
+ gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (column), cell, state_cell_func, self, NULL);
+ gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (column), cell, FALSE);
+
gtk_tree_view_append_column (GTK_TREE_VIEW (self), column);
priv->dim_label_attributes = pango_attr_list_new ();
diff --git a/src/libide/tree/meson.build b/src/libide/tree/meson.build
index 8e47f196d..b3a21d444 100644
--- a/src/libide/tree/meson.build
+++ b/src/libide/tree/meson.build
@@ -26,7 +26,11 @@ libide_tree_public_sources = [
'ide-tree-node.c',
]
-libide_tree_sources = libide_tree_public_sources
+libide_tree_private_sources = [
+ 'ide-cell-renderer-status.c',
+]
+
+libide_tree_sources = libide_tree_public_sources + libide_tree_private_sources
#
# Dependencies
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]