/* gtkshortcutsgroup.c * * Copyright (C) 2015 Christian Hergert * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library. If not, see . */ #include "config.h" #include "gtkshortcutsgroup.h" #include "gtkshortcutsshortcut.h" #include "gtkshortcutsgesture.h" #include "gtklabel.h" #include "gtkorientable.h" #include "gtksizegroup.h" #include "gtkprivate.h" #include "gtkintl.h" /** * SECTION:gtkshortcutsgroup * @Title: GtkShortcutsGroup * @Short_description: Represents a group of shortcuts in a GtkShortcutsWindow * * A GtkShortcutsGroup represents a group of related keyboard shortcuts * or gestures. The group has a title. It may optionally be associated with * a view of the application, which can be used to show only relevant shortcuts * depending on the application context. * * This widget is only meant to be used with #GtkShortcutsWindow. */ struct _GtkShortcutsGroup { GtkBox parent_instance; GtkLabel *title; gchar *view; guint height; }; struct _GtkShortcutsGroupClass { GtkBoxClass parent_class; }; G_DEFINE_TYPE (GtkShortcutsGroup, gtk_shortcuts_group, GTK_TYPE_BOX) enum { PROP_0, PROP_TITLE, PROP_VIEW, PROP_ACCEL_SIZE_GROUP, PROP_TITLE_SIZE_GROUP, PROP_HEIGHT, LAST_PROP }; static GParamSpec *properties[LAST_PROP]; static void gtk_shortcuts_group_set_accel_size_group (GtkShortcutsGroup *group, GtkSizeGroup *size_group) { GList *children, *l; children = gtk_container_get_children (GTK_CONTAINER (group)); for (l = children; l; l = l->next) { if (GTK_IS_SHORTCUTS_SHORTCUT (l->data)) g_object_set (l->data, "accel-size-group", size_group, NULL); else if (GTK_IS_SHORTCUTS_GESTURE (l->data)) g_object_set (l->data, "icon-size-group", size_group, NULL); } g_list_free (children); } static void gtk_shortcuts_group_set_title_size_group (GtkShortcutsGroup *group, GtkSizeGroup *size_group) { GList *children, *l; children = gtk_container_get_children (GTK_CONTAINER (group)); for (l = children; l; l = l->next) { if (GTK_IS_SHORTCUTS_SHORTCUT (l->data)) g_object_set (l->data, "title-size-group", size_group, NULL); else if (GTK_IS_SHORTCUTS_GESTURE (l->data)) g_object_set (l->data, "title-size-group", size_group, NULL); } g_list_free (children); } static guint gtk_shortcuts_group_get_height (GtkShortcutsGroup *group) { GList *children, *l; guint height; height = 1; children = gtk_container_get_children (GTK_CONTAINER (group)); for (l = children; l; l = l->next) { if (GTK_IS_SHORTCUTS_SHORTCUT (l->data)) height += 1; else if (GTK_IS_SHORTCUTS_GESTURE (l->data)) height += 2; } g_list_free (children); return height; } static void gtk_shortcuts_group_add (GtkContainer *container, GtkWidget *widget) { if (GTK_IS_SHORTCUTS_SHORTCUT (widget) || GTK_IS_SHORTCUTS_GESTURE (widget)) GTK_CONTAINER_CLASS (gtk_shortcuts_group_parent_class)->add (container, widget); else g_warning ("Can't add children of type %s to %s", G_OBJECT_TYPE_NAME (widget), G_OBJECT_TYPE_NAME (container)); } static void gtk_shortcuts_group_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkShortcutsGroup *self = GTK_SHORTCUTS_GROUP (object); switch (prop_id) { case PROP_TITLE: g_value_set_string (value, gtk_label_get_label (self->title)); break; case PROP_VIEW: g_value_set_string (value, self->view); break; case PROP_HEIGHT: g_value_set_uint (value, gtk_shortcuts_group_get_height (self)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void gtk_shortcuts_group_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtkShortcutsGroup *self = GTK_SHORTCUTS_GROUP (object); switch (prop_id) { case PROP_TITLE: gtk_label_set_label (self->title, g_value_get_string (value)); break; case PROP_VIEW: g_free (self->view); self->view = g_value_dup_string (value); break; case PROP_ACCEL_SIZE_GROUP: gtk_shortcuts_group_set_accel_size_group (self, GTK_SIZE_GROUP (g_value_get_object (value))); break; case PROP_TITLE_SIZE_GROUP: gtk_shortcuts_group_set_title_size_group (self, GTK_SIZE_GROUP (g_value_get_object (value))); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void gtk_shortcuts_group_finalize (GObject *object) { GtkShortcutsGroup *self = GTK_SHORTCUTS_GROUP (object); g_free (self->view); G_OBJECT_CLASS (gtk_shortcuts_group_parent_class)->finalize (object); } static void gtk_shortcuts_group_class_init (GtkShortcutsGroupClass *klass) { GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass); GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = gtk_shortcuts_group_finalize; object_class->get_property = gtk_shortcuts_group_get_property; object_class->set_property = gtk_shortcuts_group_set_property; container_class->add = gtk_shortcuts_group_add; /** * GtkShortcutsGroup:title: * * The title for this group of shortcuts. */ properties[PROP_TITLE] = g_param_spec_string ("title", P_("Title"), P_("Title"), "", (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * GtkShortcutsGroup:view: * * An optional view that the shortcuts in this group are relevant for. * The group will be hidden if the #GtkShortcutsWindow:view-name property * does not match the view of this group. * * Set this to %NULL to make the group always visible. */ properties[PROP_VIEW] = g_param_spec_string ("view", P_("View"), P_("View"), NULL, (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); /** * GtkShortcutsGroup:accel-size-group: * * The size group for the accelerator portion of shortcuts in this group. * * This is used internally by GTK+, and must not be modified by applications. */ properties[PROP_ACCEL_SIZE_GROUP] = g_param_spec_object ("accel-size-group", P_("Accelerator Size Group"), P_("Accelerator Size Group"), GTK_TYPE_SIZE_GROUP, (G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS)); /** * GtkShortcutsGroup:title-size-group: * * The size group for the textual portion of shortcuts in this group. * * This is used internally by GTK+, and must not be modified by applications. */ properties[PROP_TITLE_SIZE_GROUP] = g_param_spec_object ("title-size-group", P_("Title Size Group"), P_("Title Size Group"), GTK_TYPE_SIZE_GROUP, (G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS)); /** * GtkShortcutsGroup:height: * * A rough measure for the number of lines in this group. * * This is used internally by GTK+, and is not useful for applications. */ properties[PROP_HEIGHT] = g_param_spec_uint ("height", P_("Height"), P_("Height"), 0, G_MAXUINT, 1, (G_PARAM_READABLE | G_PARAM_STATIC_STRINGS)); g_object_class_install_properties (object_class, LAST_PROP, properties); } static void gtk_shortcuts_group_init (GtkShortcutsGroup *self) { PangoAttrList *attrs; gtk_orientable_set_orientation (GTK_ORIENTABLE (self), GTK_ORIENTATION_VERTICAL); gtk_box_set_spacing (GTK_BOX (self), 10); attrs = pango_attr_list_new (); pango_attr_list_insert (attrs, pango_attr_weight_new (PANGO_WEIGHT_BOLD)); self->title = g_object_new (GTK_TYPE_LABEL, "attributes", attrs, "visible", TRUE, "xalign", 0.0f, NULL); pango_attr_list_unref (attrs); GTK_CONTAINER_CLASS (gtk_shortcuts_group_parent_class)->add (GTK_CONTAINER (self), GTK_WIDGET (self->title)); }