/* * Copyright © 2020 Benjamin Otte * * 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. * * This library 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 library. If not, see . * * Authors: Benjamin Otte */ #include "config.h" #include "gtkboolfilter.h" #include "gtktypebuiltins.h" /** * GtkBoolFilter: * * `GtkBoolFilter` evaluates a boolean `GtkExpression` * to determine whether to include items. */ struct _GtkBoolFilter { GtkFilter parent_instance; gboolean invert; GtkExpression *expression; }; enum { PROP_0, PROP_EXPRESSION, PROP_INVERT, NUM_PROPERTIES }; G_DEFINE_TYPE (GtkBoolFilter, gtk_bool_filter, GTK_TYPE_FILTER) static GParamSpec *properties[NUM_PROPERTIES] = { NULL, }; static gboolean gtk_bool_filter_match (GtkFilter *filter, gpointer item) { GtkBoolFilter *self = GTK_BOOL_FILTER (filter); GValue value = G_VALUE_INIT; gboolean result; if (self->expression == NULL || !gtk_expression_evaluate (self->expression, item, &value)) return FALSE; result = g_value_get_boolean (&value); g_value_unset (&value); if (self->invert) result = !result; return result; } static GtkFilterMatch gtk_bool_filter_get_strictness (GtkFilter *filter) { GtkBoolFilter *self = GTK_BOOL_FILTER (filter); if (self->expression == NULL) return GTK_FILTER_MATCH_NONE; return GTK_FILTER_MATCH_SOME; } static void gtk_bool_filter_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtkBoolFilter *self = GTK_BOOL_FILTER (object); switch (prop_id) { case PROP_EXPRESSION: gtk_bool_filter_set_expression (self, gtk_value_get_expression (value)); break; case PROP_INVERT: gtk_bool_filter_set_invert (self, g_value_get_boolean (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gtk_bool_filter_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkBoolFilter *self = GTK_BOOL_FILTER (object); switch (prop_id) { case PROP_EXPRESSION: gtk_value_set_expression (value, self->expression); break; case PROP_INVERT: g_value_set_boolean (value, self->invert); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gtk_bool_filter_dispose (GObject *object) { GtkBoolFilter *self = GTK_BOOL_FILTER (object); g_clear_pointer (&self->expression, gtk_expression_unref); G_OBJECT_CLASS (gtk_bool_filter_parent_class)->dispose (object); } static void gtk_bool_filter_class_init (GtkBoolFilterClass *class) { GtkFilterClass *filter_class = GTK_FILTER_CLASS (class); GObjectClass *object_class = G_OBJECT_CLASS (class); filter_class->match = gtk_bool_filter_match; filter_class->get_strictness = gtk_bool_filter_get_strictness; object_class->get_property = gtk_bool_filter_get_property; object_class->set_property = gtk_bool_filter_set_property; object_class->dispose = gtk_bool_filter_dispose; /** * GtkBoolFilter:expression: (type GtkExpression) (attributes org.gtk.Property.get=gtk_bool_filter_get_expression org.gtk.Property.set=gtk_bool_filter_set_expression) * * The boolean expression to evaluate on item. */ properties[PROP_EXPRESSION] = gtk_param_spec_expression ("expression", NULL, NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY); /** * GtkBoolFilter:invert: (attributes org.gtk.Property.get=gtk_bool_filter_get_invert org.gtk.Property.set=gtk_bool_filter_set_invert) * * If the expression result should be inverted. */ properties[PROP_INVERT] = g_param_spec_boolean ("invert", NULL, NULL, FALSE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_EXPLICIT_NOTIFY); g_object_class_install_properties (object_class, NUM_PROPERTIES, properties); } static void gtk_bool_filter_init (GtkBoolFilter *self) { } /** * gtk_bool_filter_new: * @expression: (transfer full) (nullable): The expression to evaluate * * Creates a new bool filter. * * Returns: a new `GtkBoolFilter` */ GtkBoolFilter * gtk_bool_filter_new (GtkExpression *expression) { GtkBoolFilter *result; result = g_object_new (GTK_TYPE_BOOL_FILTER, "expression", expression, NULL); g_clear_pointer (&expression, gtk_expression_unref); return result; } /** * gtk_bool_filter_get_expression: (attributes org.gtk.Method.get_property=expression) * @self: a `GtkBoolFilter` * * Gets the expression that the filter uses to evaluate if * an item should be filtered. * * Returns: (transfer none) (nullable): a `GtkExpression` */ GtkExpression * gtk_bool_filter_get_expression (GtkBoolFilter *self) { g_return_val_if_fail (GTK_IS_BOOL_FILTER (self), NULL); return self->expression; } /** * gtk_bool_filter_set_expression: (attributes org.gtk.Method.set_property=expression) * @self: a `GtkBoolFilter` * @expression: (nullable): a `GtkExpression` * * Sets the expression that the filter uses to check if items * should be filtered. * * The expression must have a value type of %G_TYPE_BOOLEAN. */ void gtk_bool_filter_set_expression (GtkBoolFilter *self, GtkExpression *expression) { g_return_if_fail (GTK_IS_BOOL_FILTER (self)); g_return_if_fail (expression == NULL || gtk_expression_get_value_type (expression) == G_TYPE_BOOLEAN); if (self->expression == expression) return; g_clear_pointer (&self->expression, gtk_expression_unref); if (expression) self->expression = gtk_expression_ref (expression); gtk_filter_changed (GTK_FILTER (self), GTK_FILTER_CHANGE_DIFFERENT); g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_EXPRESSION]); } /** * gtk_bool_filter_get_invert: (attributes org.gtk.Method.get_property=invert) * @self: a `GtkBoolFilter` * * Returns whether the filter inverts the expression. * * Returns: %TRUE if the filter inverts */ gboolean gtk_bool_filter_get_invert (GtkBoolFilter *self) { g_return_val_if_fail (GTK_IS_BOOL_FILTER (self), TRUE); return self->invert; } /** * gtk_bool_filter_set_invert: (attributes org.gtk.Method.set_property=invert) * @self: a `GtkBoolFilter` * @invert: %TRUE to invert * * Sets whether the filter should invert the expression. */ void gtk_bool_filter_set_invert (GtkBoolFilter *self, gboolean invert) { g_return_if_fail (GTK_IS_BOOL_FILTER (self)); if (self->invert == invert) return; self->invert = invert; gtk_filter_changed (GTK_FILTER (self), GTK_FILTER_CHANGE_DIFFERENT); g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_INVERT]); }