mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-14 06:10:21 +00:00
88203fbf4d
Extract the "user filter" code from the `GtkFontChooserWidget`, for a couple of reasons: * If we want to expand the filter in the future (e.g. to filter on variable fonts, or check for multiple languages), we have a nice place to put this. * It simplifies the font chooser widget a tiny bit, as it's a pretty big file which can be hard to follow. * With a custom `GtkFilter` subclass, we can actually avoid doing a bit of work when initially showing the widget, as we can return `GTK_FILTER_MATCH_ALL` when nothing is selected yet (which is not possible with a `GtkCustomFilter'). It's not much, but it's still nice
273 lines
7.0 KiB
C
273 lines
7.0 KiB
C
/* GTK - The GIMP Toolkit
|
|
* gtkfontfilter.c:
|
|
* Copyright (C) 2024 Niels De Graef <nielsdegraef@gmail.com>
|
|
*
|
|
* 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 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <glib/gprintf.h>
|
|
#include <string.h>
|
|
|
|
#include "gtkfontfilterprivate.h"
|
|
#include "gtkprivate.h"
|
|
|
|
struct _GtkFontFilter
|
|
{
|
|
GtkFilter parent_instance;
|
|
|
|
PangoContext *pango_context;
|
|
|
|
gboolean monospace;
|
|
PangoLanguage *language;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_PANGO_CONTEXT,
|
|
PROP_MONOSPACE,
|
|
PROP_LANGUAGE,
|
|
N_PROPS
|
|
};
|
|
static GParamSpec *properties[N_PROPS] = { NULL, };
|
|
|
|
G_DEFINE_TYPE (GtkFontFilter, gtk_font_filter, GTK_TYPE_FILTER)
|
|
|
|
static void
|
|
gtk_font_filter_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkFontFilter *self = GTK_FONT_FILTER (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_PANGO_CONTEXT:
|
|
_gtk_font_filter_set_pango_context (self, g_value_get_object (value));
|
|
break;
|
|
case PROP_MONOSPACE:
|
|
_gtk_font_filter_set_monospace (self, g_value_get_boolean (value));
|
|
break;
|
|
case PROP_LANGUAGE:
|
|
_gtk_font_filter_set_language (self, g_value_get_boxed (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_font_filter_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkFontFilter *self = GTK_FONT_FILTER (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_PANGO_CONTEXT:
|
|
g_value_set_object (value, self->pango_context);
|
|
break;
|
|
case PROP_MONOSPACE:
|
|
g_value_set_boolean (value, _gtk_font_filter_get_monospace (self));
|
|
break;
|
|
case PROP_LANGUAGE:
|
|
g_value_set_boxed (value, _gtk_font_filter_get_language (self));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gtk_font_filter_match (GtkFilter *filter,
|
|
gpointer item)
|
|
{
|
|
GtkFontFilter *self = GTK_FONT_FILTER (filter);
|
|
PangoFontFamily *family;
|
|
PangoFontFace *face;
|
|
|
|
if (PANGO_IS_FONT_FAMILY (item))
|
|
{
|
|
family = item;
|
|
face = pango_font_family_get_face (family, NULL);
|
|
}
|
|
else
|
|
{
|
|
face = PANGO_FONT_FACE (item);
|
|
family = pango_font_face_get_family (face);
|
|
}
|
|
|
|
if (self->monospace &&
|
|
!pango_font_family_is_monospace (family))
|
|
return FALSE;
|
|
|
|
if (self->language)
|
|
{
|
|
PangoFontDescription *desc;
|
|
PangoFont *font;
|
|
PangoLanguage **langs;
|
|
gboolean ret = FALSE;
|
|
|
|
desc = pango_font_face_describe (face);
|
|
pango_font_description_set_size (desc, 20);
|
|
|
|
font = pango_context_load_font (self->pango_context, desc);
|
|
|
|
langs = pango_font_get_languages (font);
|
|
if (langs)
|
|
{
|
|
for (int i = 0; langs[i]; i++)
|
|
{
|
|
if (langs[i] == self->language)
|
|
{
|
|
ret = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
g_object_unref (font);
|
|
pango_font_description_free (desc);
|
|
|
|
return ret;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GtkFilterMatch
|
|
gtk_font_filter_get_strictness (GtkFilter *filter)
|
|
{
|
|
GtkFontFilter *self = GTK_FONT_FILTER (filter);
|
|
|
|
if (!self->monospace && self->language == NULL)
|
|
return GTK_FILTER_MATCH_ALL;
|
|
|
|
return GTK_FILTER_MATCH_SOME;
|
|
}
|
|
|
|
static void
|
|
gtk_font_filter_class_init (GtkFontFilterClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
GtkFilterClass *filter_class = GTK_FILTER_CLASS (klass);
|
|
|
|
filter_class->match = gtk_font_filter_match;
|
|
filter_class->get_strictness = gtk_font_filter_get_strictness;
|
|
|
|
gobject_class->set_property = gtk_font_filter_set_property;
|
|
gobject_class->get_property = gtk_font_filter_get_property;
|
|
|
|
properties[PROP_PANGO_CONTEXT] =
|
|
g_param_spec_object ("pango-context", NULL, NULL,
|
|
PANGO_TYPE_CONTEXT,
|
|
GTK_PARAM_READWRITE);
|
|
properties[PROP_MONOSPACE] =
|
|
g_param_spec_boolean ("monospace", NULL, NULL,
|
|
FALSE,
|
|
GTK_PARAM_READWRITE);
|
|
properties[PROP_LANGUAGE] =
|
|
g_param_spec_boxed ("language", NULL, NULL,
|
|
PANGO_TYPE_LANGUAGE,
|
|
GTK_PARAM_READWRITE);
|
|
|
|
g_object_class_install_properties (gobject_class, N_PROPS, properties);
|
|
}
|
|
|
|
static void
|
|
gtk_font_filter_init (GtkFontFilter *self)
|
|
{
|
|
}
|
|
|
|
void
|
|
_gtk_font_filter_set_pango_context (GtkFontFilter *self,
|
|
PangoContext *context)
|
|
{
|
|
g_return_if_fail (GTK_IS_FONT_FILTER (self));
|
|
g_return_if_fail (PANGO_IS_CONTEXT (context));
|
|
|
|
if (self->pango_context == context)
|
|
return;
|
|
|
|
self->pango_context = context;
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_PANGO_CONTEXT]);
|
|
}
|
|
|
|
gboolean
|
|
_gtk_font_filter_get_monospace (GtkFontFilter *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_FONT_FILTER (self), FALSE);
|
|
|
|
return self->monospace;
|
|
}
|
|
|
|
void
|
|
_gtk_font_filter_set_monospace (GtkFontFilter *self,
|
|
gboolean monospace)
|
|
{
|
|
g_return_if_fail (GTK_IS_FONT_FILTER (self));
|
|
|
|
if (self->monospace == monospace)
|
|
return;
|
|
|
|
self->monospace = monospace;
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_MONOSPACE]);
|
|
gtk_filter_changed (GTK_FILTER (self),
|
|
monospace ? GTK_FILTER_CHANGE_MORE_STRICT
|
|
: GTK_FILTER_CHANGE_LESS_STRICT);
|
|
}
|
|
|
|
PangoLanguage *
|
|
_gtk_font_filter_get_language (GtkFontFilter *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_FONT_FILTER (self), NULL);
|
|
|
|
return self->language;
|
|
}
|
|
|
|
void
|
|
_gtk_font_filter_set_language (GtkFontFilter *self,
|
|
PangoLanguage *lang)
|
|
{
|
|
GtkFilterChange filter_change = GTK_FILTER_CHANGE_DIFFERENT;
|
|
|
|
g_return_if_fail (GTK_IS_FONT_FILTER (self));
|
|
|
|
if (self->language == lang)
|
|
return;
|
|
|
|
if (lang == NULL || self->language == NULL)
|
|
{
|
|
filter_change = (lang != NULL) ? GTK_FILTER_CHANGE_MORE_STRICT
|
|
: GTK_FILTER_CHANGE_LESS_STRICT;
|
|
}
|
|
|
|
self->language = lang;
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_LANGUAGE]);
|
|
gtk_filter_changed (GTK_FILTER (self), filter_change);
|
|
}
|
|
|
|
GtkFilter *
|
|
_gtk_font_filter_new (void)
|
|
{
|
|
return g_object_new (GTK_TYPE_FONT_FILTER, NULL);
|
|
}
|