2018-08-04 10:14:13 +00:00
|
|
|
/*
|
|
|
|
* Copyright © 2018 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Authors: Benjamin Otte <otte@gnome.org>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "gtkshortcut.h"
|
|
|
|
|
|
|
|
#include "gtkintl.h"
|
2018-08-18 05:32:11 +00:00
|
|
|
#include "gtkshortcutaction.h"
|
2020-03-15 13:49:50 +00:00
|
|
|
#include "gtkshortcuttrigger.h"
|
2018-08-04 10:14:13 +00:00
|
|
|
#include "gtkwidget.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:gtkshortcut
|
|
|
|
* @title: GtkShortcut
|
|
|
|
* @short_description: An object describing a keyboard shortcut
|
2018-08-18 05:32:11 +00:00
|
|
|
* @see_also: #GtkShortcutController, #GtkShortcutAction,
|
|
|
|
* #GtkShortcutTrigger
|
2018-08-04 10:14:13 +00:00
|
|
|
*
|
|
|
|
* GtkShortcut is the low level object used for managing keyboard
|
|
|
|
* shortcuts.
|
|
|
|
*
|
|
|
|
* It contains a description of how to trigger the shortcut via a
|
|
|
|
* #GtkShortcutTrigger and a way to activate the shortcut on a widget
|
2018-08-18 05:32:11 +00:00
|
|
|
* via #GtkShortcutAction.
|
2018-08-04 10:14:13 +00:00
|
|
|
*
|
|
|
|
* The actual work is usually done via #GtkShortcutController, which
|
|
|
|
* decides if and when to activate a shortcut. Using that controller
|
2018-08-18 05:32:11 +00:00
|
|
|
* directly however is rarely necessary as various higher level
|
2018-08-04 10:14:13 +00:00
|
|
|
* convenience APIs exist on #GtkWidgets that make it easier to use
|
|
|
|
* shortcuts in GTK.
|
|
|
|
*
|
|
|
|
* #GtkShortcut does provide functionality to make it easy for users
|
|
|
|
* to work with shortcuts, either by providing informational strings
|
|
|
|
* for display purposes or by allowing shortcuts to be configured.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct _GtkShortcut
|
|
|
|
{
|
|
|
|
GObject parent_instance;
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
GtkShortcutAction *action;
|
2020-03-15 13:49:50 +00:00
|
|
|
GtkShortcutTrigger *trigger;
|
2018-08-04 10:14:13 +00:00
|
|
|
GVariant *args;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
2020-02-05 17:08:14 +00:00
|
|
|
PROP_ACTION,
|
2018-08-18 05:32:11 +00:00
|
|
|
PROP_ARGUMENTS,
|
2020-03-15 13:49:50 +00:00
|
|
|
PROP_TRIGGER,
|
2018-08-04 10:14:13 +00:00
|
|
|
|
|
|
|
N_PROPS
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (GtkShortcut, gtk_shortcut, G_TYPE_OBJECT)
|
|
|
|
|
|
|
|
static GParamSpec *properties[N_PROPS] = { NULL, };
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_shortcut_dispose (GObject *object)
|
|
|
|
{
|
|
|
|
GtkShortcut *self = GTK_SHORTCUT (object);
|
|
|
|
|
2020-03-20 15:19:45 +00:00
|
|
|
g_clear_object (&self->action);
|
2020-03-19 15:57:02 +00:00
|
|
|
g_clear_object (&self->trigger);
|
2018-08-04 10:14:13 +00:00
|
|
|
g_clear_pointer (&self->args, g_variant_unref);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_shortcut_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_shortcut_get_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkShortcut *self = GTK_SHORTCUT (object);
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
2018-08-18 05:32:11 +00:00
|
|
|
case PROP_ACTION:
|
|
|
|
g_value_set_boxed (value, self->action);
|
2018-08-04 10:14:13 +00:00
|
|
|
break;
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
case PROP_ARGUMENTS:
|
|
|
|
g_value_set_variant (value, self->args);
|
2020-02-05 17:08:14 +00:00
|
|
|
break;
|
|
|
|
|
2020-03-15 13:49:50 +00:00
|
|
|
case PROP_TRIGGER:
|
2020-03-19 15:57:02 +00:00
|
|
|
g_value_set_object (value, self->trigger);
|
2020-03-15 13:49:50 +00:00
|
|
|
break;
|
|
|
|
|
2018-08-04 10:14:13 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_shortcut_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkShortcut *self = GTK_SHORTCUT (object);
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
2018-08-18 05:32:11 +00:00
|
|
|
case PROP_ACTION:
|
2020-03-20 15:19:45 +00:00
|
|
|
gtk_shortcut_set_action (self, g_value_dup_object (value));
|
2018-08-04 10:14:13 +00:00
|
|
|
break;
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
case PROP_ARGUMENTS:
|
|
|
|
gtk_shortcut_set_arguments (self, g_value_get_variant (value));
|
2020-02-05 17:08:14 +00:00
|
|
|
break;
|
|
|
|
|
2020-03-15 13:49:50 +00:00
|
|
|
case PROP_TRIGGER:
|
2020-03-19 15:57:02 +00:00
|
|
|
gtk_shortcut_set_trigger (self, g_value_dup_object (value));
|
2020-03-15 13:49:50 +00:00
|
|
|
break;
|
|
|
|
|
2018-08-04 10:14:13 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_shortcut_class_init (GtkShortcutClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
gobject_class->dispose = gtk_shortcut_dispose;
|
|
|
|
gobject_class->get_property = gtk_shortcut_get_property;
|
|
|
|
gobject_class->set_property = gtk_shortcut_set_property;
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
/**
|
|
|
|
* GtkShortcut:action:
|
|
|
|
*
|
|
|
|
* The action that gets activated by this shortcut.
|
|
|
|
*/
|
|
|
|
properties[PROP_ACTION] =
|
2020-03-20 15:19:45 +00:00
|
|
|
g_param_spec_object ("action",
|
|
|
|
P_("Action"),
|
|
|
|
P_("The action activated by this shortcut"),
|
|
|
|
GTK_TYPE_SHORTCUT_ACTION,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_EXPLICIT_NOTIFY |
|
|
|
|
G_PARAM_STATIC_STRINGS);
|
2018-08-18 05:32:11 +00:00
|
|
|
|
2018-08-04 10:14:13 +00:00
|
|
|
/**
|
|
|
|
* GtkShortcut:arguments:
|
|
|
|
*
|
|
|
|
* Arguments passed to activation.
|
|
|
|
*/
|
|
|
|
properties[PROP_ARGUMENTS] =
|
|
|
|
g_param_spec_variant ("arguments",
|
|
|
|
P_("Arguments"),
|
|
|
|
P_("Arguments passed to activation"),
|
|
|
|
G_VARIANT_TYPE_ANY,
|
|
|
|
NULL,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
|
2020-03-15 13:49:50 +00:00
|
|
|
/**
|
|
|
|
* GtkShortcut:trigger:
|
|
|
|
*
|
|
|
|
* The trigger that triggers this shortcut.
|
|
|
|
*/
|
|
|
|
properties[PROP_TRIGGER] =
|
2020-03-19 15:57:02 +00:00
|
|
|
g_param_spec_object ("trigger",
|
|
|
|
P_("Trigger"),
|
|
|
|
P_("The trigger for this shortcut"),
|
|
|
|
GTK_TYPE_SHORTCUT_TRIGGER,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_EXPLICIT_NOTIFY |
|
|
|
|
G_PARAM_STATIC_STRINGS);
|
2020-03-15 13:49:50 +00:00
|
|
|
|
2018-08-04 10:14:13 +00:00
|
|
|
g_object_class_install_properties (gobject_class, N_PROPS, properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_shortcut_init (GtkShortcut *self)
|
|
|
|
{
|
2020-03-20 15:19:45 +00:00
|
|
|
self->action = g_object_ref (gtk_nothing_action_get ());
|
2020-03-19 15:57:02 +00:00
|
|
|
self->trigger = g_object_ref (gtk_never_trigger_get ());
|
2018-08-04 10:14:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_shortcut_new:
|
2020-03-19 15:57:02 +00:00
|
|
|
* @trigger: (transfer full) (nullable): The trigger that will trigger the shortcut
|
|
|
|
* @action: (transfer full) (nullable): The action that will be activated upon
|
2018-08-18 18:01:43 +00:00
|
|
|
* triggering
|
2018-08-04 10:14:13 +00:00
|
|
|
*
|
2018-08-18 18:01:43 +00:00
|
|
|
* Creates a new #GtkShortcut that is triggered by @trigger and then activates
|
|
|
|
* @action.
|
2018-08-04 10:14:13 +00:00
|
|
|
*
|
|
|
|
* Returns: a new #GtkShortcut
|
|
|
|
**/
|
|
|
|
GtkShortcut *
|
2018-08-18 18:01:43 +00:00
|
|
|
gtk_shortcut_new (GtkShortcutTrigger *trigger,
|
|
|
|
GtkShortcutAction *action)
|
2018-08-04 10:14:13 +00:00
|
|
|
{
|
2018-08-18 18:01:43 +00:00
|
|
|
GtkShortcut *shortcut;
|
|
|
|
|
|
|
|
shortcut = g_object_new (GTK_TYPE_SHORTCUT,
|
|
|
|
"action", action,
|
|
|
|
"trigger", trigger,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (trigger)
|
2020-03-19 15:57:02 +00:00
|
|
|
g_object_unref (trigger);
|
2018-08-18 18:01:43 +00:00
|
|
|
if (action)
|
2020-03-20 15:19:45 +00:00
|
|
|
g_object_unref (action);
|
2018-08-18 18:01:43 +00:00
|
|
|
|
|
|
|
return shortcut;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_shortcut_new_with_arguments: (skip)
|
2020-03-19 15:57:02 +00:00
|
|
|
* @trigger: (transfer full) (nullable): The trigger that will trigger the shortcut
|
|
|
|
* @action: (transfer full) (nullable): The action that will be activated upon
|
2018-08-18 18:01:43 +00:00
|
|
|
* triggering
|
|
|
|
* @format_string: (allow-none): GVariant format string for arguments or %NULL for
|
|
|
|
* no arguments
|
|
|
|
* @...: arguments, as given by format string.
|
|
|
|
*
|
|
|
|
* Creates a new #GtkShortcut that is triggered by @trigger and then activates
|
|
|
|
* @action with arguments given by @format_string.
|
|
|
|
*
|
|
|
|
* Returns: a new #GtkShortcut
|
|
|
|
**/
|
|
|
|
GtkShortcut *
|
|
|
|
gtk_shortcut_new_with_arguments (GtkShortcutTrigger *trigger,
|
|
|
|
GtkShortcutAction *action,
|
|
|
|
const gchar *format_string,
|
|
|
|
...)
|
|
|
|
{
|
|
|
|
GtkShortcut *shortcut;
|
|
|
|
GVariant *args;
|
|
|
|
|
|
|
|
if (format_string)
|
|
|
|
{
|
|
|
|
va_list valist;
|
|
|
|
va_start (valist, format_string);
|
|
|
|
args = g_variant_new_va (format_string, NULL, &valist);
|
|
|
|
va_end (valist);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
args = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
shortcut = g_object_new (GTK_TYPE_SHORTCUT,
|
|
|
|
"action", action,
|
|
|
|
"arguments", args,
|
|
|
|
"trigger", trigger,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (trigger)
|
2020-03-19 15:57:02 +00:00
|
|
|
g_object_unref (trigger);
|
2018-08-18 18:01:43 +00:00
|
|
|
if (action)
|
2020-03-20 15:19:45 +00:00
|
|
|
g_object_unref (action);
|
2018-08-18 18:01:43 +00:00
|
|
|
|
|
|
|
return shortcut;
|
2018-08-04 10:14:13 +00:00
|
|
|
}
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
/**
|
|
|
|
* gtk_shortcut_get_action:
|
|
|
|
* @self: a #GtkShortcut
|
|
|
|
*
|
|
|
|
* Gets the action that is activated by this shortcut.
|
|
|
|
*
|
|
|
|
* Returns: (transfer none): the action
|
|
|
|
**/
|
|
|
|
GtkShortcutAction *
|
|
|
|
gtk_shortcut_get_action (GtkShortcut *self)
|
2018-08-12 20:07:27 +00:00
|
|
|
{
|
2018-08-18 05:32:11 +00:00
|
|
|
g_return_val_if_fail (GTK_IS_SHORTCUT (self), NULL);
|
2018-08-12 20:07:27 +00:00
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
return self->action;
|
2018-08-12 20:07:27 +00:00
|
|
|
}
|
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
/**
|
|
|
|
* gtk_shortcut_set_action:
|
|
|
|
* @self: a #GtkShortcut
|
|
|
|
* @action: (transfer full) (nullable): The new action.
|
|
|
|
* If the @action is %NULL, the nothing action will be used.
|
|
|
|
*
|
|
|
|
* Sets the new action for @self to be @action.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_shortcut_set_action (GtkShortcut *self,
|
|
|
|
GtkShortcutAction *action)
|
2018-08-12 20:07:27 +00:00
|
|
|
{
|
2018-08-18 05:32:11 +00:00
|
|
|
g_return_if_fail (GTK_IS_SHORTCUT (self));
|
2018-08-12 20:07:27 +00:00
|
|
|
|
2018-08-18 05:32:11 +00:00
|
|
|
if (action == NULL)
|
2020-03-20 15:19:45 +00:00
|
|
|
action = g_object_ref (gtk_nothing_action_get ());
|
2018-08-12 20:07:27 +00:00
|
|
|
|
2020-03-20 15:19:45 +00:00
|
|
|
if (g_set_object (&self->action, action))
|
2018-08-12 20:07:27 +00:00
|
|
|
{
|
2020-03-20 15:19:45 +00:00
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_ACTION]);
|
|
|
|
g_object_unref (action);
|
2018-08-12 20:07:27 +00:00
|
|
|
}
|
2018-08-04 10:14:13 +00:00
|
|
|
}
|
|
|
|
|
2020-03-15 13:49:50 +00:00
|
|
|
/**
|
|
|
|
* gtk_shortcut_get_trigger:
|
|
|
|
* @self: a #GtkShortcut
|
|
|
|
*
|
|
|
|
* Gets the trigger used to trigger @self.
|
|
|
|
*
|
|
|
|
* Returns: (transfer none): the trigger used
|
|
|
|
**/
|
|
|
|
GtkShortcutTrigger *
|
|
|
|
gtk_shortcut_get_trigger (GtkShortcut *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_SHORTCUT (self), NULL);
|
|
|
|
|
|
|
|
return self->trigger;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_shortcut_set_trigger:
|
|
|
|
* @self: a #GtkShortcut
|
|
|
|
* @trigger: (transfer full) (nullable): The new trigger.
|
|
|
|
* If the @trigger is %NULL, the never trigger will be used.
|
|
|
|
*
|
|
|
|
* Sets the new trigger for @self to be @trigger.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
gtk_shortcut_set_trigger (GtkShortcut *self,
|
|
|
|
GtkShortcutTrigger *trigger)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_SHORTCUT (self));
|
|
|
|
|
|
|
|
if (trigger == NULL)
|
2020-03-19 15:57:02 +00:00
|
|
|
trigger = g_object_ref (gtk_never_trigger_get ());
|
2020-03-15 13:49:50 +00:00
|
|
|
|
2020-03-19 15:57:02 +00:00
|
|
|
if (g_set_object (&self->trigger, trigger))
|
2020-03-15 13:49:50 +00:00
|
|
|
{
|
2020-03-19 15:57:02 +00:00
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_TRIGGER]);
|
|
|
|
g_object_unref (trigger);
|
2020-03-15 13:49:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-04 10:14:13 +00:00
|
|
|
GVariant *
|
|
|
|
gtk_shortcut_get_arguments (GtkShortcut *self)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_SHORTCUT (self), NULL);
|
|
|
|
|
|
|
|
return self->args;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gtk_shortcut_set_arguments (GtkShortcut *self,
|
|
|
|
GVariant *args)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_SHORTCUT (self));
|
|
|
|
|
|
|
|
if (self->args == args)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_clear_pointer (&self->args, g_variant_unref);
|
|
|
|
if (args)
|
|
|
|
self->args = g_variant_ref_sink (args);
|
|
|
|
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_ARGUMENTS]);
|
|
|
|
}
|