426 lines
10 KiB
C
426 lines
10 KiB
C
/*
|
|
* 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
|
|
* 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 "gtksearchengineprivate.h"
|
|
#include "gtksearchenginemodelprivate.h"
|
|
#include "gtksearchenginequartzprivate.h"
|
|
#include <glib/gi18n-lib.h>
|
|
#include "gtkprivate.h"
|
|
|
|
#include <gdk/gdk.h> /* for GDK_WINDOWING_MACOS */
|
|
|
|
#if defined(HAVE_TRACKER3)
|
|
#include "gtksearchenginetracker3private.h"
|
|
#endif
|
|
|
|
struct _GtkSearchEnginePrivate {
|
|
GtkSearchEngine *native;
|
|
gboolean native_running;
|
|
gboolean got_results;
|
|
char *native_error;
|
|
|
|
GtkSearchEngine *model;
|
|
gboolean model_running;
|
|
char *model_error;
|
|
|
|
gboolean running;
|
|
gboolean recursive;
|
|
GHashTable *hits;
|
|
|
|
GtkQuery *query;
|
|
};
|
|
|
|
enum
|
|
{
|
|
HITS_ADDED,
|
|
FINISHED,
|
|
ERROR,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
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)
|
|
{
|
|
g_set_object (&engine->priv->query, query);
|
|
|
|
if (engine->priv->native)
|
|
_gtk_search_engine_set_query (engine->priv->native, query);
|
|
|
|
if (engine->priv->model)
|
|
_gtk_search_engine_set_query (engine->priv->model, 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->model)
|
|
{
|
|
g_clear_pointer (&engine->priv->model_error, g_free);
|
|
_gtk_search_engine_start (engine->priv->model);
|
|
engine->priv->model_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->model)
|
|
{
|
|
_gtk_search_engine_stop (engine->priv->model);
|
|
engine->priv->model_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->model);
|
|
g_free (engine->priv->model_error);
|
|
|
|
g_clear_pointer (&engine->priv->hits, g_hash_table_unref);
|
|
|
|
g_clear_object (&engine->priv->query);
|
|
|
|
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 (I_("hits-added"),
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GtkSearchEngineClass, hits_added),
|
|
NULL, NULL,
|
|
NULL,
|
|
G_TYPE_NONE, 1,
|
|
G_TYPE_POINTER);
|
|
|
|
signals[FINISHED] =
|
|
g_signal_new (I_("finished"),
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GtkSearchEngineClass, finished),
|
|
NULL, NULL,
|
|
NULL,
|
|
G_TYPE_NONE, 1, G_TYPE_BOOLEAN);
|
|
|
|
signals[ERROR] =
|
|
g_signal_new (I_("error"),
|
|
G_TYPE_FROM_CLASS (class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GtkSearchEngineClass, error),
|
|
NULL, NULL,
|
|
NULL,
|
|
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;
|
|
GtkSearchHit *hit;
|
|
|
|
added = NULL;
|
|
|
|
for (l = hits; l; l = l->next)
|
|
{
|
|
hit = l->data;
|
|
|
|
if (!g_hash_table_contains (composite->priv->hits, hit))
|
|
{
|
|
hit = _gtk_search_hit_dup (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;
|
|
|
|
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->model_error)
|
|
_gtk_search_engine_error (engine, engine->priv->model_error);
|
|
else
|
|
_gtk_search_engine_finished (engine, engine->priv->got_results);
|
|
|
|
engine->priv->got_results = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
finished (GtkSearchEngine *engine,
|
|
gboolean got_results,
|
|
gpointer data)
|
|
{
|
|
GtkSearchEngine *composite = GTK_SEARCH_ENGINE (data);
|
|
|
|
if (engine == composite->priv->native)
|
|
composite->priv->native_running = FALSE;
|
|
else if (engine == composite->priv->model)
|
|
composite->priv->model_running = FALSE;
|
|
|
|
composite->priv->got_results |= got_results;
|
|
update_status (composite);
|
|
}
|
|
|
|
static void
|
|
error (GtkSearchEngine *engine,
|
|
const char *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->model)
|
|
{
|
|
g_free (composite->priv->model_error);
|
|
composite->priv->model_error = g_strdup (message);
|
|
composite->priv->model_running = FALSE;
|
|
}
|
|
|
|
update_status (composite);
|
|
}
|
|
|
|
static gboolean
|
|
search_hit_equal (gconstpointer a, gconstpointer b)
|
|
{
|
|
const GtkSearchHit *ha = (const GtkSearchHit *)a;
|
|
const GtkSearchHit *hb = (const GtkSearchHit *)b;
|
|
|
|
return g_file_equal (ha->file, hb->file);
|
|
}
|
|
|
|
|
|
static guint
|
|
search_hit_hash (gconstpointer a)
|
|
{
|
|
const GtkSearchHit *ha = (const GtkSearchHit *)a;
|
|
|
|
return g_file_hash (ha->file);
|
|
}
|
|
|
|
GtkSearchHit *
|
|
_gtk_search_hit_dup (GtkSearchHit *hit)
|
|
{
|
|
GtkSearchHit *dup;
|
|
|
|
dup = g_new (GtkSearchHit, 1);
|
|
dup->file = g_object_ref (hit->file);
|
|
if (hit->info)
|
|
dup->info = g_object_ref (hit->info);
|
|
else
|
|
dup->info = NULL;
|
|
|
|
return dup;
|
|
}
|
|
|
|
void
|
|
_gtk_search_hit_free (GtkSearchHit *hit)
|
|
{
|
|
g_clear_object (&hit->file);
|
|
g_clear_object (&hit->info);
|
|
g_free (hit);
|
|
}
|
|
|
|
static void
|
|
connect_engine_signals (GtkSearchEngine *engine,
|
|
gpointer data)
|
|
{
|
|
g_signal_connect_object (engine, "hits-added", G_CALLBACK (hits_added), data, 0);
|
|
g_signal_connect_object (engine, "finished", G_CALLBACK (finished), data, 0);
|
|
g_signal_connect_object (engine, "error", G_CALLBACK (error), data, 0);
|
|
}
|
|
|
|
GtkSearchEngine *
|
|
_gtk_search_engine_new (void)
|
|
{
|
|
GtkSearchEngine *engine;
|
|
|
|
engine = g_object_new (GTK_TYPE_SEARCH_ENGINE, NULL);
|
|
|
|
#if defined(HAVE_TRACKER3)
|
|
engine->priv->native = gtk_search_engine_tracker3_new ();
|
|
if (engine->priv->native)
|
|
{
|
|
g_debug ("Using Tracker3 search engine");
|
|
connect_engine_signals (engine->priv->native, engine);
|
|
}
|
|
#endif
|
|
|
|
#ifdef GDK_WINDOWING_MACOS
|
|
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->hits = g_hash_table_new_full (search_hit_hash, search_hit_equal,
|
|
(GDestroyNotify)_gtk_search_hit_free, NULL);
|
|
|
|
return engine;
|
|
}
|
|
|
|
void
|
|
_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);
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
GTK_SEARCH_ENGINE_GET_CLASS (engine)->stop (engine);
|
|
}
|
|
|
|
void
|
|
_gtk_search_engine_hits_added (GtkSearchEngine *engine,
|
|
GList *hits)
|
|
{
|
|
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
|
|
|
|
g_signal_emit (engine, signals[HITS_ADDED], 0, hits);
|
|
}
|
|
|
|
void
|
|
_gtk_search_engine_finished (GtkSearchEngine *engine,
|
|
gboolean got_results)
|
|
{
|
|
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
|
|
|
|
g_signal_emit (engine, signals[FINISHED], 0, got_results);
|
|
}
|
|
|
|
void
|
|
_gtk_search_engine_error (GtkSearchEngine *engine,
|
|
const char *error_message)
|
|
{
|
|
g_return_if_fail (GTK_IS_SEARCH_ENGINE (engine));
|
|
|
|
g_signal_emit (engine, signals[ERROR], 0, error_message);
|
|
}
|
|
|
|
void
|
|
_gtk_search_engine_set_model (GtkSearchEngine *engine,
|
|
GtkFileSystemModel *model)
|
|
{
|
|
g_clear_object (&engine->priv->model);
|
|
if (model)
|
|
{
|
|
engine->priv->model = _gtk_search_engine_model_new (model);
|
|
connect_engine_signals (engine->priv->model, engine);
|
|
if (engine->priv->query)
|
|
_gtk_search_engine_set_query (engine->priv->model, engine->priv->query);
|
|
}
|
|
}
|