2012-06-01 14:04:31 +00:00
|
|
|
|
/* GTK - The GIMP Toolkit
|
|
|
|
|
* Copyright (C) 2012 Red Hat, Inc.
|
|
|
|
|
*
|
|
|
|
|
* Authors:
|
|
|
|
|
* - Bastien Nocera <bnocera@redhat.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/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Modified by the GTK+ Team and others 2012. See the AUTHORS
|
|
|
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
|
|
|
* files for a list of changes. These files are distributed with
|
|
|
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
2016-10-15 20:22:48 +00:00
|
|
|
|
#include "gtksearchentryprivate.h"
|
2015-10-29 14:55:28 +00:00
|
|
|
|
|
|
|
|
|
#include "gtkaccessible.h"
|
2014-11-25 03:40:45 +00:00
|
|
|
|
#include "gtkbindings.h"
|
2019-02-17 04:57:18 +00:00
|
|
|
|
#include "gtkeditable.h"
|
|
|
|
|
#include "gtkbox.h"
|
|
|
|
|
#include "gtkgesturemultipress.h"
|
|
|
|
|
#include "gtktextprivate.h"
|
|
|
|
|
#include "gtkimage.h"
|
2015-10-29 14:55:28 +00:00
|
|
|
|
#include "gtkintl.h"
|
2019-02-17 04:57:18 +00:00
|
|
|
|
#include "gtkprivate.h"
|
2015-10-29 14:55:28 +00:00
|
|
|
|
#include "gtkmarshalers.h"
|
|
|
|
|
#include "gtkstylecontext.h"
|
2018-03-11 12:53:17 +00:00
|
|
|
|
#include "gtkeventcontrollerkey.h"
|
2019-02-22 20:50:45 +00:00
|
|
|
|
#include "a11y/gtkentryaccessible.h"
|
|
|
|
|
|
2012-06-01 14:04:31 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* SECTION:gtksearchentry
|
|
|
|
|
* @Short_description: An entry which shows a search icon
|
|
|
|
|
* @Title: GtkSearchEntry
|
|
|
|
|
*
|
2014-11-25 03:40:45 +00:00
|
|
|
|
* #GtkSearchEntry is a subclass of #GtkEntry that has been
|
|
|
|
|
* tailored for use as a search entry.
|
2012-06-01 14:04:31 +00:00
|
|
|
|
*
|
2014-11-25 03:40:45 +00:00
|
|
|
|
* It will show an inactive symbolic “find” icon when the search
|
|
|
|
|
* entry is empty, and a symbolic “clear” icon when there is text.
|
|
|
|
|
* Clicking on the “clear” icon will empty the search entry.
|
2012-06-01 14:04:31 +00:00
|
|
|
|
*
|
|
|
|
|
* Note that the search/clear icon is shown using a secondary
|
|
|
|
|
* icon, and thus does not work if you are using the secondary
|
|
|
|
|
* icon position for some other purpose.
|
|
|
|
|
*
|
2013-07-29 01:49:37 +00:00
|
|
|
|
* To make filtering appear more reactive, it is a good idea to
|
|
|
|
|
* not react to every change in the entry text immediately, but
|
|
|
|
|
* only after a short delay. To support this, #GtkSearchEntry
|
|
|
|
|
* emits the #GtkSearchEntry::search-changed signal which can
|
|
|
|
|
* be used instead of the #GtkEditable::changed signal.
|
|
|
|
|
*
|
2014-11-25 03:40:45 +00:00
|
|
|
|
* The #GtkSearchEntry::previous-match, #GtkSearchEntry::next-match
|
2017-01-31 09:39:39 +00:00
|
|
|
|
* and #GtkSearchEntry::stop-search signals can be used to implement
|
2014-11-25 03:40:45 +00:00
|
|
|
|
* moving between search results and ending the search.
|
|
|
|
|
*
|
|
|
|
|
* Often, GtkSearchEntry will be fed events by means of being
|
2015-04-26 21:49:28 +00:00
|
|
|
|
* placed inside a #GtkSearchBar. If that is not the case,
|
2018-03-11 12:53:17 +00:00
|
|
|
|
* you can use gtk_search_entry_set_key_capture_widget() to let it
|
|
|
|
|
* capture key input from another widget.
|
2012-06-01 14:04:31 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
enum {
|
2019-02-17 04:57:18 +00:00
|
|
|
|
ACTIVATE,
|
2013-07-29 01:49:37 +00:00
|
|
|
|
SEARCH_CHANGED,
|
2014-11-25 03:40:45 +00:00
|
|
|
|
NEXT_MATCH,
|
|
|
|
|
PREVIOUS_MATCH,
|
|
|
|
|
STOP_SEARCH,
|
2013-07-29 01:49:37 +00:00
|
|
|
|
LAST_SIGNAL
|
|
|
|
|
};
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
enum {
|
|
|
|
|
PROP_0,
|
|
|
|
|
PROP_PLACEHOLDER_TEXT,
|
|
|
|
|
PROP_ACTIVATES_DEFAULT,
|
|
|
|
|
NUM_PROPERTIES,
|
|
|
|
|
};
|
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
static GParamSpec *props[NUM_PROPERTIES] = { NULL, };
|
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
typedef struct {
|
2018-03-11 12:53:17 +00:00
|
|
|
|
GtkWidget *capture_widget;
|
|
|
|
|
GtkEventController *capture_widget_controller;
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkWidget *box;
|
|
|
|
|
GtkWidget *entry;
|
|
|
|
|
GtkWidget *icon;
|
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
guint delayed_changed_id;
|
2014-11-25 03:40:45 +00:00
|
|
|
|
gboolean content_changed;
|
|
|
|
|
gboolean search_stopped;
|
2013-05-16 16:20:19 +00:00
|
|
|
|
} GtkSearchEntryPrivate;
|
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
static void gtk_search_entry_editable_init (GtkEditableInterface *iface);
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkSearchEntry, gtk_search_entry, GTK_TYPE_WIDGET,
|
2013-07-29 01:49:37 +00:00
|
|
|
|
G_ADD_PRIVATE (GtkSearchEntry)
|
|
|
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_EDITABLE,
|
|
|
|
|
gtk_search_entry_editable_init))
|
2013-06-27 19:02:52 +00:00
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
/* 150 mseconds of delay */
|
|
|
|
|
#define DELAYED_TIMEOUT_ID 150
|
|
|
|
|
|
2014-11-25 03:40:45 +00:00
|
|
|
|
static void
|
2019-02-17 04:57:18 +00:00
|
|
|
|
text_changed (GtkSearchEntry *entry)
|
2014-11-25 03:40:45 +00:00
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
|
|
|
|
priv->content_changed = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_search_entry_finalize (GObject *object)
|
2014-11-25 03:40:45 +00:00
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (object);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_editable_finish_delegate (GTK_EDITABLE (entry));
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
g_clear_pointer (&priv->box, gtk_widget_unparent);
|
2013-05-16 16:20:19 +00:00
|
|
|
|
|
|
|
|
|
if (priv->delayed_changed_id > 0)
|
|
|
|
|
g_source_remove (priv->delayed_changed_id);
|
|
|
|
|
|
2018-03-11 12:53:17 +00:00
|
|
|
|
gtk_search_entry_set_key_capture_widget (GTK_SEARCH_ENTRY (object), NULL);
|
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
G_OBJECT_CLASS (gtk_search_entry_parent_class)->finalize (object);
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-25 03:40:45 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_stop_search (GtkSearchEntry *entry)
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
|
|
|
|
priv->search_stopped = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_set_property (GObject *object,
|
|
|
|
|
guint prop_id,
|
|
|
|
|
const GValue *value,
|
|
|
|
|
GParamSpec *pspec)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (object);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
if (gtk_editable_delegate_set_property (object, prop_id, value, pspec))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
|
{
|
|
|
|
|
case PROP_PLACEHOLDER_TEXT:
|
|
|
|
|
gtk_text_set_placeholder_text (GTK_TEXT (priv->entry), g_value_get_string (value));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case PROP_ACTIVATES_DEFAULT:
|
|
|
|
|
gtk_text_set_activates_default (GTK_TEXT (priv->entry), g_value_get_boolean (value));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_get_property (GObject *object,
|
|
|
|
|
guint prop_id,
|
|
|
|
|
GValue *value,
|
|
|
|
|
GParamSpec *pspec)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (object);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
if (gtk_editable_delegate_get_property (object, prop_id, value, pspec))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
|
{
|
|
|
|
|
case PROP_PLACEHOLDER_TEXT:
|
|
|
|
|
g_value_set_string (value, gtk_text_get_placeholder_text (GTK_TEXT (priv->entry)));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case PROP_ACTIVATES_DEFAULT:
|
|
|
|
|
g_value_set_boolean (value, gtk_text_get_activates_default (GTK_TEXT (priv->entry)));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_measure (GtkWidget *widget,
|
|
|
|
|
GtkOrientation orientation,
|
|
|
|
|
int for_size,
|
|
|
|
|
int *minimum,
|
|
|
|
|
int *natural,
|
|
|
|
|
int *minimum_baseline,
|
|
|
|
|
int *natural_baseline)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (widget);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
gtk_widget_measure (priv->box, orientation, for_size,
|
|
|
|
|
minimum, natural,
|
|
|
|
|
minimum_baseline, natural_baseline);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_size_allocate (GtkWidget *widget,
|
|
|
|
|
int width,
|
|
|
|
|
int height,
|
|
|
|
|
int baseline)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (widget);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
gtk_widget_size_allocate (priv->box,
|
|
|
|
|
&(GtkAllocation) { 0, 0, width, height },
|
|
|
|
|
baseline);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-22 19:12:52 +00:00
|
|
|
|
static AtkObject *
|
|
|
|
|
gtk_search_entry_get_accessible (GtkWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
AtkObject *atk_obj;
|
|
|
|
|
|
|
|
|
|
atk_obj = GTK_WIDGET_CLASS (gtk_search_entry_parent_class)->get_accessible (widget);
|
|
|
|
|
atk_object_set_name (atk_obj, _("Search"));
|
|
|
|
|
|
|
|
|
|
return atk_obj;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_grab_focus (GtkWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (widget);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
gtk_text_grab_focus_without_selecting (GTK_TEXT (priv->entry));
|
|
|
|
|
}
|
|
|
|
|
|
2012-06-01 14:04:31 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_class_init (GtkSearchEntryClass *klass)
|
|
|
|
|
{
|
2013-05-16 16:20:19 +00:00
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
GtkBindingSet *binding_set;
|
2013-05-16 16:20:19 +00:00
|
|
|
|
|
|
|
|
|
object_class->finalize = gtk_search_entry_finalize;
|
2019-02-17 04:57:18 +00:00
|
|
|
|
object_class->get_property = gtk_search_entry_get_property;
|
|
|
|
|
object_class->set_property = gtk_search_entry_set_property;
|
|
|
|
|
|
|
|
|
|
widget_class->measure = gtk_search_entry_measure;
|
|
|
|
|
widget_class->size_allocate = gtk_search_entry_size_allocate;
|
2019-02-22 19:12:52 +00:00
|
|
|
|
widget_class->get_accessible = gtk_search_entry_get_accessible;
|
2019-02-17 04:57:18 +00:00
|
|
|
|
widget_class->grab_focus = gtk_search_entry_grab_focus;
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
|
|
|
|
klass->stop_search = gtk_search_entry_stop_search;
|
2012-06-01 14:04:31 +00:00
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
props[PROP_PLACEHOLDER_TEXT] =
|
|
|
|
|
g_param_spec_string ("placeholder-text",
|
|
|
|
|
P_("Placeholder text"),
|
|
|
|
|
P_("Show text in the entry when it’s empty and unfocused"),
|
|
|
|
|
NULL,
|
|
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
|
|
|
|
|
props[PROP_ACTIVATES_DEFAULT] =
|
|
|
|
|
g_param_spec_boolean ("activates-default",
|
|
|
|
|
P_("Activates default"),
|
|
|
|
|
P_("Whether to activate the default widget (such as the default button in a dialog) when Enter is pressed"),
|
|
|
|
|
FALSE,
|
|
|
|
|
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY);
|
|
|
|
|
|
|
|
|
|
g_object_class_install_properties (object_class, NUM_PROPERTIES, props);
|
|
|
|
|
gtk_editable_install_properties (object_class, NUM_PROPERTIES);
|
|
|
|
|
|
|
|
|
|
signals[ACTIVATE] =
|
|
|
|
|
g_signal_new (I_("activate"),
|
|
|
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
|
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
|
|
|
G_STRUCT_OFFSET (GtkSearchEntryClass, activate),
|
|
|
|
|
NULL, NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
G_TYPE_NONE, 0);
|
2013-07-29 01:49:37 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GtkSearchEntry::search-changed:
|
|
|
|
|
* @entry: the entry on which the signal was emitted
|
|
|
|
|
*
|
|
|
|
|
* The #GtkSearchEntry::search-changed signal is emitted with a short
|
|
|
|
|
* delay of 150 milliseconds after the last change to the entry text.
|
|
|
|
|
*/
|
|
|
|
|
signals[SEARCH_CHANGED] =
|
|
|
|
|
g_signal_new (I_("search-changed"),
|
|
|
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
|
|
|
G_SIGNAL_RUN_LAST,
|
|
|
|
|
G_STRUCT_OFFSET (GtkSearchEntryClass, search_changed),
|
|
|
|
|
NULL, NULL,
|
2016-08-29 14:00:17 +00:00
|
|
|
|
NULL,
|
2013-07-29 01:49:37 +00:00
|
|
|
|
G_TYPE_NONE, 0);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GtkSearchEntry::next-match:
|
|
|
|
|
* @entry: the entry on which the signal was emitted
|
|
|
|
|
*
|
|
|
|
|
* The ::next-match signal is a [keybinding signal][GtkBindingSignal]
|
|
|
|
|
* which gets emitted when the user initiates a move to the next match
|
|
|
|
|
* for the current search string.
|
|
|
|
|
*
|
|
|
|
|
* Applications should connect to it, to implement moving between
|
|
|
|
|
* matches.
|
|
|
|
|
*
|
|
|
|
|
* The default bindings for this signal is Ctrl-g.
|
|
|
|
|
*/
|
|
|
|
|
signals[NEXT_MATCH] =
|
|
|
|
|
g_signal_new (I_("next-match"),
|
|
|
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
|
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
|
|
|
G_STRUCT_OFFSET (GtkSearchEntryClass, next_match),
|
|
|
|
|
NULL, NULL,
|
2016-08-29 14:00:17 +00:00
|
|
|
|
NULL,
|
2014-11-25 03:40:45 +00:00
|
|
|
|
G_TYPE_NONE, 0);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GtkSearchEntry::previous-match:
|
|
|
|
|
* @entry: the entry on which the signal was emitted
|
|
|
|
|
*
|
|
|
|
|
* The ::previous-match signal is a [keybinding signal][GtkBindingSignal]
|
|
|
|
|
* which gets emitted when the user initiates a move to the previous match
|
|
|
|
|
* for the current search string.
|
|
|
|
|
*
|
|
|
|
|
* Applications should connect to it, to implement moving between
|
|
|
|
|
* matches.
|
|
|
|
|
*
|
|
|
|
|
* The default bindings for this signal is Ctrl-Shift-g.
|
|
|
|
|
*/
|
|
|
|
|
signals[PREVIOUS_MATCH] =
|
|
|
|
|
g_signal_new (I_("previous-match"),
|
|
|
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
|
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
|
|
|
G_STRUCT_OFFSET (GtkSearchEntryClass, previous_match),
|
|
|
|
|
NULL, NULL,
|
2016-08-29 14:00:17 +00:00
|
|
|
|
NULL,
|
2014-11-25 03:40:45 +00:00
|
|
|
|
G_TYPE_NONE, 0);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GtkSearchEntry::stop-search:
|
|
|
|
|
* @entry: the entry on which the signal was emitted
|
|
|
|
|
*
|
|
|
|
|
* The ::stop-search signal is a [keybinding signal][GtkBindingSignal]
|
|
|
|
|
* which gets emitted when the user stops a search via keyboard input.
|
|
|
|
|
*
|
|
|
|
|
* Applications should connect to it, to implement hiding the search
|
|
|
|
|
* entry in this case.
|
|
|
|
|
*
|
|
|
|
|
* The default bindings for this signal is Escape.
|
|
|
|
|
*/
|
|
|
|
|
signals[STOP_SEARCH] =
|
|
|
|
|
g_signal_new (I_("stop-search"),
|
|
|
|
|
G_OBJECT_CLASS_TYPE (object_class),
|
|
|
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
|
|
|
G_STRUCT_OFFSET (GtkSearchEntryClass, stop_search),
|
|
|
|
|
NULL, NULL,
|
2016-08-29 14:00:17 +00:00
|
|
|
|
NULL,
|
2014-11-25 03:40:45 +00:00
|
|
|
|
G_TYPE_NONE, 0);
|
|
|
|
|
|
|
|
|
|
binding_set = gtk_binding_set_by_class (klass);
|
|
|
|
|
|
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_g, GDK_CONTROL_MASK,
|
|
|
|
|
"next-match", 0);
|
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_g, GDK_SHIFT_MASK | GDK_CONTROL_MASK,
|
|
|
|
|
"previous-match", 0);
|
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_Escape, 0,
|
|
|
|
|
"stop-search", 0);
|
2019-02-17 04:57:18 +00:00
|
|
|
|
|
2019-02-22 20:50:45 +00:00
|
|
|
|
gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_ENTRY_ACCESSIBLE);
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_widget_class_set_css_name (widget_class, I_("entry"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static GtkEditable *
|
|
|
|
|
gtk_search_entry_get_delegate (GtkEditable *editable)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = GTK_SEARCH_ENTRY (editable);
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
return GTK_EDITABLE (priv->entry);
|
2013-07-29 01:49:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_editable_init (GtkEditableInterface *iface)
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
iface->get_delegate = gtk_search_entry_get_delegate;
|
2013-07-29 01:49:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_search_entry_icon_release (GtkGestureMultiPress *press,
|
|
|
|
|
int n_press,
|
|
|
|
|
double x,
|
|
|
|
|
double y,
|
|
|
|
|
GtkSearchEntry *entry)
|
2013-07-29 01:49:37 +00:00
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
gtk_editable_set_text (GTK_EDITABLE (priv->entry), "");
|
2012-06-01 14:04:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
static gboolean
|
|
|
|
|
gtk_search_entry_changed_timeout_cb (gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntry *entry = user_data;
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2013-05-16 16:20:19 +00:00
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
g_signal_emit (entry, signals[SEARCH_CHANGED], 0);
|
2013-05-16 16:20:19 +00:00
|
|
|
|
priv->delayed_changed_id = 0;
|
|
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
reset_timeout (GtkSearchEntry *entry)
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2013-05-16 16:20:19 +00:00
|
|
|
|
|
|
|
|
|
if (priv->delayed_changed_id > 0)
|
|
|
|
|
g_source_remove (priv->delayed_changed_id);
|
|
|
|
|
priv->delayed_changed_id = g_timeout_add (DELAYED_TIMEOUT_ID,
|
|
|
|
|
gtk_search_entry_changed_timeout_cb,
|
|
|
|
|
entry);
|
2019-02-05 10:26:20 +00:00
|
|
|
|
g_source_set_name_by_id (priv->delayed_changed_id, "[gtk] gtk_search_entry_changed_timeout_cb");
|
2013-05-16 16:20:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2012-06-01 14:04:31 +00:00
|
|
|
|
static void
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_search_entry_changed (GtkEditable *editable,
|
|
|
|
|
GtkSearchEntry *entry)
|
2012-06-01 14:04:31 +00:00
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
const char *str;
|
2012-06-01 14:04:31 +00:00
|
|
|
|
|
2013-05-16 16:20:19 +00:00
|
|
|
|
/* Update the icons first */
|
2019-02-17 04:57:18 +00:00
|
|
|
|
str = gtk_editable_get_text (GTK_EDITABLE (priv->entry));
|
2012-06-01 14:04:31 +00:00
|
|
|
|
|
|
|
|
|
if (str == NULL || *str == '\0')
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_widget_hide (priv->icon);
|
2013-05-16 16:20:19 +00:00
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
if (priv->delayed_changed_id > 0)
|
|
|
|
|
{
|
|
|
|
|
g_source_remove (priv->delayed_changed_id);
|
|
|
|
|
priv->delayed_changed_id = 0;
|
|
|
|
|
}
|
|
|
|
|
g_signal_emit (entry, signals[SEARCH_CHANGED], 0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_widget_show (priv->icon);
|
|
|
|
|
|
2013-07-29 01:49:37 +00:00
|
|
|
|
/* Queue up the timeout */
|
|
|
|
|
reset_timeout (entry);
|
|
|
|
|
}
|
2012-06-01 14:04:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
static void
|
|
|
|
|
notify_cb (GObject *object,
|
|
|
|
|
GParamSpec *pspec,
|
|
|
|
|
gpointer data)
|
|
|
|
|
{
|
|
|
|
|
/* The editable interface properties are already forwarded by the editable delegate setup */
|
|
|
|
|
if (g_str_equal (pspec->name, "placeholder-text") ||
|
|
|
|
|
g_str_equal (pspec->name, "activates-default"))
|
|
|
|
|
g_object_notify (data, pspec->name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
activate_cb (GtkText *text,
|
|
|
|
|
gpointer data)
|
|
|
|
|
{
|
|
|
|
|
g_signal_emit (data, signals[ACTIVATE], 0);
|
|
|
|
|
}
|
|
|
|
|
|
2012-06-01 14:04:31 +00:00
|
|
|
|
static void
|
|
|
|
|
gtk_search_entry_init (GtkSearchEntry *entry)
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
GtkGesture *press;
|
|
|
|
|
|
|
|
|
|
gtk_widget_set_has_surface (GTK_WIDGET (entry), FALSE);
|
|
|
|
|
|
|
|
|
|
priv->box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
|
gtk_widget_set_parent (priv->box, GTK_WIDGET (entry));
|
2019-02-22 20:55:05 +00:00
|
|
|
|
gtk_widget_set_hexpand (priv->box, FALSE);
|
|
|
|
|
gtk_widget_set_vexpand (priv->box, FALSE);
|
2019-02-17 04:57:18 +00:00
|
|
|
|
|
|
|
|
|
priv->entry = gtk_text_new ();
|
|
|
|
|
gtk_widget_set_hexpand (priv->entry, TRUE);
|
|
|
|
|
gtk_widget_set_vexpand (priv->entry, TRUE);
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (priv->box), GTK_WIDGET (priv->entry));
|
|
|
|
|
gtk_editable_init_delegate (GTK_EDITABLE (entry));
|
|
|
|
|
g_signal_connect_swapped (priv->entry, "changed", G_CALLBACK (text_changed), entry);
|
|
|
|
|
g_signal_connect_after (priv->entry, "changed", G_CALLBACK (gtk_search_entry_changed), entry);
|
|
|
|
|
g_signal_connect_swapped (priv->entry, "preedit-changed", G_CALLBACK (text_changed), entry);
|
|
|
|
|
g_signal_connect (priv->entry, "notify", G_CALLBACK (notify_cb), entry);
|
|
|
|
|
g_signal_connect (priv->entry, "activate", G_CALLBACK (activate_cb), entry);
|
|
|
|
|
|
|
|
|
|
priv->icon = gtk_image_new_from_icon_name ("edit-clear-symbolic");
|
2019-02-19 05:32:35 +00:00
|
|
|
|
gtk_widget_set_tooltip_text (priv->icon, _("Clear entry"));
|
2019-02-17 04:57:18 +00:00
|
|
|
|
gtk_container_add (GTK_CONTAINER (priv->box), GTK_WIDGET (priv->icon));
|
|
|
|
|
gtk_widget_hide (priv->icon);
|
2013-08-15 13:20:19 +00:00
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
press = gtk_gesture_multi_press_new ();
|
|
|
|
|
g_signal_connect (press, "released", G_CALLBACK (gtk_search_entry_icon_release), entry);
|
|
|
|
|
gtk_widget_add_controller (priv->icon, GTK_EVENT_CONTROLLER (press));
|
2013-08-15 13:20:19 +00:00
|
|
|
|
|
2017-11-18 13:18:11 +00:00
|
|
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (GTK_WIDGET (entry)), I_("search"));
|
2012-06-01 14:04:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_search_entry_new:
|
|
|
|
|
*
|
|
|
|
|
* Creates a #GtkSearchEntry, with a find icon when the search field is
|
|
|
|
|
* empty, and a clear icon when it isn't.
|
|
|
|
|
*
|
2014-02-19 23:49:43 +00:00
|
|
|
|
* Returns: a new #GtkSearchEntry
|
2012-06-01 14:04:31 +00:00
|
|
|
|
*/
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtk_search_entry_new (void)
|
|
|
|
|
{
|
|
|
|
|
return GTK_WIDGET (g_object_new (GTK_TYPE_SEARCH_ENTRY, NULL));
|
|
|
|
|
}
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
2014-12-27 23:29:07 +00:00
|
|
|
|
gboolean
|
2018-03-11 12:53:17 +00:00
|
|
|
|
gtk_search_entry_is_keynav (guint keyval,
|
|
|
|
|
GdkModifierType state)
|
2014-11-25 03:40:45 +00:00
|
|
|
|
{
|
|
|
|
|
if (keyval == GDK_KEY_Tab || keyval == GDK_KEY_KP_Tab ||
|
|
|
|
|
keyval == GDK_KEY_Up || keyval == GDK_KEY_KP_Up ||
|
|
|
|
|
keyval == GDK_KEY_Down || keyval == GDK_KEY_KP_Down ||
|
|
|
|
|
keyval == GDK_KEY_Left || keyval == GDK_KEY_KP_Left ||
|
|
|
|
|
keyval == GDK_KEY_Right || keyval == GDK_KEY_KP_Right ||
|
|
|
|
|
keyval == GDK_KEY_Home || keyval == GDK_KEY_KP_Home ||
|
|
|
|
|
keyval == GDK_KEY_End || keyval == GDK_KEY_KP_End ||
|
|
|
|
|
keyval == GDK_KEY_Page_Up || keyval == GDK_KEY_KP_Page_Up ||
|
|
|
|
|
keyval == GDK_KEY_Page_Down || keyval == GDK_KEY_KP_Page_Down ||
|
|
|
|
|
((state & (GDK_CONTROL_MASK | GDK_MOD1_MASK)) != 0))
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
|
|
/* Other navigation events should get automatically
|
|
|
|
|
* ignored as they will not change the content of the entry
|
|
|
|
|
*/
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_search_entry_handle_event:
|
|
|
|
|
* @entry: a #GtkSearchEntry
|
|
|
|
|
* @event: a key event
|
|
|
|
|
*
|
|
|
|
|
* This function should be called when the top-level window
|
|
|
|
|
* which contains the search entry received a key event. If
|
|
|
|
|
* the entry is part of a #GtkSearchBar, it is preferable
|
|
|
|
|
* to call gtk_search_bar_handle_event() instead, which will
|
|
|
|
|
* reveal the entry in addition to passing the event to this
|
|
|
|
|
* function.
|
|
|
|
|
*
|
|
|
|
|
* If the key event is handled by the search entry and starts
|
|
|
|
|
* or continues a search, %GDK_EVENT_STOP will be returned.
|
|
|
|
|
* The caller should ensure that the entry is shown in this
|
|
|
|
|
* case, and not propagate the event further.
|
|
|
|
|
*
|
|
|
|
|
* Returns: %GDK_EVENT_STOP if the key press event resulted
|
|
|
|
|
* in a search beginning or continuing, %GDK_EVENT_PROPAGATE
|
|
|
|
|
* otherwise.
|
|
|
|
|
*/
|
|
|
|
|
gboolean
|
|
|
|
|
gtk_search_entry_handle_event (GtkSearchEntry *entry,
|
|
|
|
|
GdkEvent *event)
|
|
|
|
|
{
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
2014-11-25 03:40:45 +00:00
|
|
|
|
gboolean handled;
|
2018-03-11 12:53:17 +00:00
|
|
|
|
guint keyval, state;
|
2014-11-25 03:40:45 +00:00
|
|
|
|
|
|
|
|
|
if (!gtk_widget_get_realized (GTK_WIDGET (entry)))
|
|
|
|
|
gtk_widget_realize (GTK_WIDGET (entry));
|
|
|
|
|
|
2017-08-25 14:46:01 +00:00
|
|
|
|
gdk_event_get_keyval (event, &keyval);
|
2018-03-11 12:53:17 +00:00
|
|
|
|
gdk_event_get_state (event, &state);
|
2017-08-25 14:46:01 +00:00
|
|
|
|
|
2018-03-11 12:53:17 +00:00
|
|
|
|
if (gtk_search_entry_is_keynav (keyval, state) ||
|
2017-08-25 14:46:01 +00:00
|
|
|
|
keyval == GDK_KEY_space ||
|
|
|
|
|
keyval == GDK_KEY_Menu)
|
2014-11-25 03:40:45 +00:00
|
|
|
|
return GDK_EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
|
|
priv->content_changed = FALSE;
|
|
|
|
|
priv->search_stopped = FALSE;
|
|
|
|
|
|
|
|
|
|
handled = gtk_widget_event (GTK_WIDGET (entry), event);
|
|
|
|
|
|
|
|
|
|
return handled && priv->content_changed && !priv->search_stopped ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE;
|
|
|
|
|
}
|
2018-03-11 12:53:17 +00:00
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
capture_widget_key_handled (GtkEventControllerKey *controller,
|
|
|
|
|
guint keyval,
|
|
|
|
|
guint keycode,
|
|
|
|
|
GdkModifierType state,
|
|
|
|
|
GtkWidget *entry)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (GTK_SEARCH_ENTRY (entry));
|
|
|
|
|
gboolean handled;
|
|
|
|
|
|
|
|
|
|
if (gtk_search_entry_is_keynav (keyval, state) ||
|
|
|
|
|
keyval == GDK_KEY_space ||
|
|
|
|
|
keyval == GDK_KEY_Menu)
|
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
|
|
priv->content_changed = FALSE;
|
|
|
|
|
priv->search_stopped = FALSE;
|
|
|
|
|
|
2019-02-19 12:51:37 +00:00
|
|
|
|
handled = gtk_event_controller_key_forward (controller, priv->entry);
|
2018-03-11 12:53:17 +00:00
|
|
|
|
|
|
|
|
|
return handled && priv->content_changed && !priv->search_stopped ? GDK_EVENT_STOP : GDK_EVENT_PROPAGATE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_search_entry_set_key_capture_widget:
|
2018-04-06 14:20:48 +00:00
|
|
|
|
* @entry: a #GtkSearchEntry
|
2018-03-11 12:53:17 +00:00
|
|
|
|
* @widget: (nullable) (transfer none): a #GtkWidget
|
|
|
|
|
*
|
|
|
|
|
* Sets @widget as the widget that @entry will capture key events from.
|
|
|
|
|
*
|
|
|
|
|
* Key events are consumed by the search entry to start or
|
|
|
|
|
* continue a search.
|
|
|
|
|
*
|
|
|
|
|
* If the entry is part of a #GtkSearchBar, it is preferable
|
|
|
|
|
* to call gtk_search_bar_set_key_capture_widget() instead, which
|
|
|
|
|
* will reveal the entry in addition to triggering the search entry.
|
|
|
|
|
**/
|
|
|
|
|
void
|
|
|
|
|
gtk_search_entry_set_key_capture_widget (GtkSearchEntry *entry,
|
|
|
|
|
GtkWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_SEARCH_ENTRY (entry));
|
|
|
|
|
g_return_if_fail (!widget || GTK_IS_WIDGET (widget));
|
|
|
|
|
|
|
|
|
|
if (priv->capture_widget)
|
|
|
|
|
{
|
2018-04-20 17:58:06 +00:00
|
|
|
|
gtk_widget_remove_controller (priv->capture_widget,
|
|
|
|
|
priv->capture_widget_controller);
|
2018-03-11 12:53:17 +00:00
|
|
|
|
g_object_remove_weak_pointer (G_OBJECT (priv->capture_widget),
|
|
|
|
|
(gpointer *) &priv->capture_widget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
priv->capture_widget = widget;
|
|
|
|
|
|
|
|
|
|
if (widget)
|
|
|
|
|
{
|
|
|
|
|
g_object_add_weak_pointer (G_OBJECT (priv->capture_widget),
|
|
|
|
|
(gpointer *) &priv->capture_widget);
|
|
|
|
|
|
2018-04-20 17:58:06 +00:00
|
|
|
|
priv->capture_widget_controller = gtk_event_controller_key_new ();
|
2018-03-11 12:53:17 +00:00
|
|
|
|
gtk_event_controller_set_propagation_phase (priv->capture_widget_controller,
|
|
|
|
|
GTK_PHASE_CAPTURE);
|
|
|
|
|
g_signal_connect (priv->capture_widget_controller, "key-pressed",
|
|
|
|
|
G_CALLBACK (capture_widget_key_handled), entry);
|
|
|
|
|
g_signal_connect (priv->capture_widget_controller, "key-released",
|
|
|
|
|
G_CALLBACK (capture_widget_key_handled), entry);
|
2018-04-20 17:58:06 +00:00
|
|
|
|
gtk_widget_add_controller (widget, priv->capture_widget_controller);
|
2018-03-11 12:53:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* gtk_search_entry_get_key_capture_widget:
|
|
|
|
|
* @entry: a #GtkSearchEntry
|
|
|
|
|
*
|
|
|
|
|
* Gets the widget that @entry is capturing key events from.
|
|
|
|
|
*
|
2018-04-06 14:20:48 +00:00
|
|
|
|
* Returns: (transfer none): The key capture widget.
|
2018-03-11 12:53:17 +00:00
|
|
|
|
**/
|
|
|
|
|
GtkWidget *
|
|
|
|
|
gtk_search_entry_get_key_capture_widget (GtkSearchEntry *entry)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
g_return_val_if_fail (GTK_IS_SEARCH_ENTRY (entry), NULL);
|
|
|
|
|
|
|
|
|
|
return priv->capture_widget;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-17 04:57:18 +00:00
|
|
|
|
GtkEventController *
|
|
|
|
|
gtk_search_entry_get_key_controller (GtkSearchEntry *entry)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
|
|
|
|
return gtk_text_get_key_controller (GTK_TEXT (priv->entry));
|
|
|
|
|
}
|
2019-02-20 15:29:26 +00:00
|
|
|
|
|
|
|
|
|
GtkText *
|
|
|
|
|
gtk_search_entry_get_text_widget (GtkSearchEntry *entry)
|
|
|
|
|
{
|
|
|
|
|
GtkSearchEntryPrivate *priv = gtk_search_entry_get_instance_private (entry);
|
|
|
|
|
|
2019-02-20 18:10:09 +00:00
|
|
|
|
return GTK_TEXT (priv->entry);
|
2019-02-20 15:29:26 +00:00
|
|
|
|
}
|