gtk2/gtk/gtksearchengine.c

382 lines
9.7 KiB
C
Raw Normal View History

/*
* Copyright (C) 2005 Novell, Inc.
*
* 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
2012-02-27 13:01:10 +00:00
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*
* Author: Anders Carlsson <andersca@imendio.com>
*
* Based on nautilus-search-engine.c
*/
#include "config.h"
#include "gtksearchengine.h"
#include "gtksearchenginesimple.h"
#include "gtksearchenginetracker.h"
#include "gtksearchenginequartz.h"
#include <gdk/gdk.h> /* for GDK_WINDOWING_QUARTZ */
2011-11-19 22:57:08 +00:00
#ifndef G_OS_WIN32 /* No tracker on Windows */
#define HAVE_TRACKER 1
#endif
struct _GtkSearchEnginePrivate {
GtkSearchEngine *native;
gboolean native_running;
gchar *native_error;
GtkSearchEngine *simple;
gboolean simple_running;
gchar *simple_error;
gboolean running;
GHashTable *hits;
};
2015-02-28 21:13:18 +00:00
enum
{
HITS_ADDED,
HITS_SUBTRACTED,
FINISHED,
ERROR,
LAST_SIGNAL
2015-02-28 21:13:18 +00:00
};
static guint signals[LAST_SIGNAL];
G_DEFINE_TYPE_WITH_PRIVATE (GtkSearchEngine, _gtk_search_engine, G_TYPE_OBJECT);
static void
set_query (GtkSearchEngine *engine,
GtkQuery *query)
{
if (engine->priv->native)
_gtk_search_engine_set_query (engine->priv->native, query);
if (engine->priv->simple)
_gtk_search_engine_set_query (engine->priv->simple, query);
}
static void
start (GtkSearchEngine *engine)
{
g_hash_table_remove_all (engine->priv->hits);
if (engine->priv->native)
{
g_clear_pointer (&engine->priv->native_error, g_free);
_gtk_search_engine_start (engine->priv->native);
engine->priv->native_running = TRUE;
}
if (engine->priv->simple)
{
g_clear_pointer (&engine->priv->simple_error, g_free);
_gtk_search_engine_start (engine->priv->simple);
engine->priv->simple_running = TRUE;
}
engine->priv->running = TRUE;
}
static void
stop (GtkSearchEngine *engine)
{
if (engine->priv->native)
{
_gtk_search_engine_stop (engine->priv->native);
engine->priv->native_running = FALSE;
}
if (engine->priv->simple)
{
_gtk_search_engine_stop (engine->priv->simple);
engine->priv->simple_running = FALSE;
}
engine->priv->running = FALSE;
g_hash_table_remove_all (engine->priv->hits);
}
static void
finalize (GObject *object)
{
GtkSearchEngine *engine = GTK_SEARCH_ENGINE (object);
g_clear_object (&engine->priv->native);
g_free (engine->priv->native_error);
g_clear_object (&engine->priv->simple);
g_free (engine->priv->simple_error);
g_clear_pointer (&engine->priv->hits, g_hash_table_unref);
G_OBJECT_CLASS (_gtk_search_engine_parent_class)->finalize (object);
}
static void
_gtk_search_engine_class_init (GtkSearchEngineClass *class)
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
object_class->finalize = finalize;
class->set_query = set_query;
class->start = start;
class->stop = stop;
signals[HITS_ADDED] =
g_signal_new ("hits-added",
2015-02-28 21:13:18 +00:00
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (GtkSearchEngineClass, hits_added),
NULL, NULL,
g_cclosure_marshal_VOID__POINTER,
G_TYPE_NONE, 1,
G_TYPE_POINTER);
signals[HITS_SUBTRACTED] =
g_signal_new ("hits-subtracted",
2015-02-28 21:13:18 +00:00
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (GtkSearchEngineClass, hits_subtracted),
NULL, NULL,
g_cclosure_marshal_VOID__POINTER,
G_TYPE_NONE, 1,
G_TYPE_POINTER);
signals[FINISHED] =
g_signal_new ("finished",
2015-02-28 21:13:18 +00:00
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (GtkSearchEngineClass, finished),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
signals[ERROR] =
g_signal_new ("error",
2015-02-28 21:13:18 +00:00
G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (GtkSearchEngineClass, error),
NULL, NULL,
g_cclosure_marshal_VOID__STRING,
G_TYPE_NONE, 1,
G_TYPE_STRING);
}
static void
_gtk_search_engine_init (GtkSearchEngine *engine)
{
engine->priv = _gtk_search_engine_get_instance_private (engine);
}
static void
hits_added (GtkSearchEngine *engine,
GList *hits,
gpointer data)
{
GtkSearchEngine *composite = GTK_SEARCH_ENGINE (data);
GList *added, *l;
added = NULL;
for (l = hits; l; l = l->next)
{
gchar *hit = l->data;
if (!g_hash_table_contains (composite->priv->hits, hit))
{
hit = g_strdup (hit);
g_hash_table_add (composite->priv->hits, hit);
added = g_list_prepend (added, hit);
}
}
if (added)
{
_gtk_search_engine_hits_added (composite, added);
g_list_free (added);
}
}
static void
update_status (GtkSearchEngine *engine)
{
gboolean running;
running = engine->priv->native_running || engine->priv->simple_running;
if (running != engine->priv->running)
{
engine->priv->running = running;
if (!running)
{
if (engine->priv->native_error)
_gtk_search_engine_error (engine, engine->priv->native_error);
else if (engine->priv->simple_error)
_gtk_search_engine_error (engine, engine->priv->simple_error);
else
_gtk_search_engine_finished (engine);
}
}
}
static void
finished (GtkSearchEngine *engine,
gpointer data)
{
GtkSearchEngine *composite = GTK_SEARCH_ENGINE (data);
if (engine == composite->priv->native)
composite->priv->native_running = FALSE;
else if (engine == composite->priv->simple)
composite->priv->simple_running = FALSE;
update_status (composite);
}
static void
error (GtkSearchEngine *engine,
const gchar *message,
gpointer data)
{
GtkSearchEngine *composite = GTK_SEARCH_ENGINE (data);
if (engine == composite->priv->native)
{
g_free (composite->priv->native_error);
composite->priv->native_error = g_strdup (message);
composite->priv->native_running = FALSE;
}
else if (engine == composite->priv->simple)
{
g_free (composite->priv->native_error);
composite->priv->native_error = g_strdup (message);
composite->priv->simple_running = FALSE;
}
update_status (composite);
}
static void
connect_engine_signals (GtkSearchEngine *engine,
gpointer data)
{
g_signal_connect (engine, "hits-added", G_CALLBACK (hits_added), data);
g_signal_connect (engine, "finished", G_CALLBACK (finished), data);
g_signal_connect (engine, "error", G_CALLBACK (error), data);
}
GtkSearchEngine *
_gtk_search_engine_new (void)
{
GtkSearchEngine *engine;
engine = g_object_new (GTK_TYPE_SEARCH_ENGINE, NULL);
2015-02-28 21:13:18 +00:00
#ifdef HAVE_TRACKER
engine->priv->native = _gtk_search_engine_tracker_new ();
if (engine->priv->native)
{
g_debug ("Using Tracker search engine");
connect_engine_signals (engine->priv->native, engine);
}
#endif
2015-02-28 21:13:18 +00:00
#ifdef GDK_WINDOWING_QUARTZ
engine->priv->native = _gtk_search_engine_quartz_new ();
if (engine->priv->native)
{
g_debug ("Using Quartz search engine");
connect_engine_signals (engine->priv->native, engine);
}
#endif
engine->priv->simple = _gtk_search_engine_simple_new ();
g_debug ("Using simple search engine");
connect_engine_signals (engine->priv->simple, engine);
engine->priv->hits = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
return engine;
}
void
2015-02-28 21:13:18 +00:00
_gtk_search_engine_set_query (GtkSearchEngine *engine,
GtkQuery *query)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
g_return_if_fail (GTK_SEARCH_ENGINE_GET_CLASS (engine)->set_query != NULL);
2015-02-28 21:13:18 +00:00
GTK_SEARCH_ENGINE_GET_CLASS (engine)->set_query (engine, query);
}
void
_gtk_search_engine_start (GtkSearchEngine *engine)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
g_return_if_fail (GTK_SEARCH_ENGINE_GET_CLASS (engine)->start != NULL);
2015-02-28 21:13:18 +00:00
GTK_SEARCH_ENGINE_GET_CLASS (engine)->start (engine);
}
void
_gtk_search_engine_stop (GtkSearchEngine *engine)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
g_return_if_fail (GTK_SEARCH_ENGINE_GET_CLASS (engine)->stop != NULL);
2015-02-28 21:13:18 +00:00
GTK_SEARCH_ENGINE_GET_CLASS (engine)->stop (engine);
}
2015-02-28 21:13:18 +00:00
void
_gtk_search_engine_hits_added (GtkSearchEngine *engine,
GList *hits)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
2015-02-28 21:13:18 +00:00
g_signal_emit (engine, signals[HITS_ADDED], 0, hits);
}
2015-02-28 21:13:18 +00:00
void
_gtk_search_engine_hits_subtracted (GtkSearchEngine *engine,
GList *hits)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
2015-02-28 21:13:18 +00:00
g_signal_emit (engine, signals[HITS_SUBTRACTED], 0, hits);
}
2015-02-28 21:13:18 +00:00
void
_gtk_search_engine_finished (GtkSearchEngine *engine)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
2015-02-28 21:13:18 +00:00
g_signal_emit (engine, signals[FINISHED], 0);
}
void
2015-02-28 21:13:18 +00:00
_gtk_search_engine_error (GtkSearchEngine *engine,
const gchar *error_message)
{
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
2015-02-28 21:13:18 +00:00
g_signal_emit (engine, signals[ERROR], 0, error_message);
}