/*
* Copyright (C) 2005 Red Hat, Inc
*
* Nautilus is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* Nautilus 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
* General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program; see the file COPYING. If not,
* see .
*
* Author: Alexander Larsson
*
*/
#include
#include "nautilus-search-hit.h"
#include "nautilus-search-provider.h"
#include "nautilus-search-engine-model.h"
#include "nautilus-directory.h"
#include "nautilus-directory-private.h"
#include "nautilus-file.h"
#include "nautilus-ui-utilities.h"
#define DEBUG_FLAG NAUTILUS_DEBUG_SEARCH
#include "nautilus-debug.h"
#include
#include
#include
struct _NautilusSearchEngineModel
{
GObject parent;
NautilusQuery *query;
GList *hits;
NautilusDirectory *directory;
gboolean query_pending;
guint finished_id;
};
enum
{
PROP_0,
PROP_RUNNING,
LAST_PROP
};
static void nautilus_search_provider_init (NautilusSearchProviderInterface *iface);
G_DEFINE_TYPE_WITH_CODE (NautilusSearchEngineModel,
nautilus_search_engine_model,
G_TYPE_OBJECT,
G_IMPLEMENT_INTERFACE (NAUTILUS_TYPE_SEARCH_PROVIDER,
nautilus_search_provider_init))
static void
finalize (GObject *object)
{
NautilusSearchEngineModel *model;
model = NAUTILUS_SEARCH_ENGINE_MODEL (object);
if (model->hits != NULL)
{
g_list_free_full (model->hits, g_object_unref);
model->hits = NULL;
}
if (model->finished_id != 0)
{
g_source_remove (model->finished_id);
model->finished_id = 0;
}
g_clear_object (&model->directory);
g_clear_object (&model->query);
G_OBJECT_CLASS (nautilus_search_engine_model_parent_class)->finalize (object);
}
static gboolean
search_finished (NautilusSearchEngineModel *model)
{
model->finished_id = 0;
if (model->hits != NULL)
{
DEBUG ("Model engine hits added");
nautilus_search_provider_hits_added (NAUTILUS_SEARCH_PROVIDER (model),
model->hits);
g_list_free_full (model->hits, g_object_unref);
model->hits = NULL;
}
model->query_pending = FALSE;
g_object_notify (G_OBJECT (model), "running");
DEBUG ("Model engine finished");
nautilus_search_provider_finished (NAUTILUS_SEARCH_PROVIDER (model),
NAUTILUS_SEARCH_PROVIDER_STATUS_NORMAL);
g_object_unref (model);
return FALSE;
}
static void
search_finished_idle (NautilusSearchEngineModel *model)
{
if (model->finished_id != 0)
{
return;
}
model->finished_id = g_idle_add ((GSourceFunc) search_finished, model);
}
static void
model_directory_ready_cb (NautilusDirectory *directory,
GList *list,
gpointer user_data)
{
NautilusSearchEngineModel *model = user_data;
g_autoptr (GPtrArray) mime_types = NULL;
gchar *uri, *display_name;
GList *files, *hits, *l;
NautilusFile *file;
gdouble match;
gboolean found;
NautilusSearchHit *hit;
GDateTime *initial_date;
GDateTime *end_date;
GPtrArray *date_range;
files = nautilus_directory_get_file_list (directory);
mime_types = nautilus_query_get_mime_types (model->query);
hits = NULL;
for (l = files; l != NULL; l = l->next)
{
g_autoptr (GDateTime) mtime = NULL;
g_autoptr (GDateTime) atime = NULL;
g_autoptr (GDateTime) ctime = NULL;
file = l->data;
display_name = nautilus_file_get_display_name (file);
match = nautilus_query_matches_string (model->query, display_name);
found = (match > -1);
if (found && mime_types->len > 0)
{
found = FALSE;
for (gint i = 0; i < mime_types->len; i++)
{
if (nautilus_file_is_mime_type (file, g_ptr_array_index (mime_types, i)))
{
found = TRUE;
break;
}
}
}
mtime = g_date_time_new_from_unix_local (nautilus_file_get_mtime (file));
atime = g_date_time_new_from_unix_local (nautilus_file_get_atime (file));
ctime = g_date_time_new_from_unix_local (nautilus_file_get_btime (file));
date_range = nautilus_query_get_date_range (model->query);
if (found && date_range != NULL)
{
NautilusQuerySearchType type;
GDateTime *target_date;
type = nautilus_query_get_search_type (model->query);
initial_date = g_ptr_array_index (date_range, 0);
end_date = g_ptr_array_index (date_range, 1);
switch (type)
{
case NAUTILUS_QUERY_SEARCH_TYPE_LAST_ACCESS:
{
target_date = atime;
}
break;
case NAUTILUS_QUERY_SEARCH_TYPE_LAST_MODIFIED:
{
target_date = mtime;
}
break;
case NAUTILUS_QUERY_SEARCH_TYPE_CREATED:
{
target_date = ctime;
}
break;
default:
{
target_date = NULL;
}
}
found = nautilus_date_time_is_between_dates (target_date,
initial_date,
end_date);
g_ptr_array_unref (date_range);
}
if (found)
{
uri = nautilus_file_get_uri (file);
hit = nautilus_search_hit_new (uri);
nautilus_search_hit_set_fts_rank (hit, match);
nautilus_search_hit_set_modification_time (hit, mtime);
nautilus_search_hit_set_access_time (hit, atime);
nautilus_search_hit_set_creation_time (hit, ctime);
hits = g_list_prepend (hits, hit);
g_free (uri);
}
g_free (display_name);
}
nautilus_file_list_free (files);
model->hits = hits;
search_finished (model);
}
static void
nautilus_search_engine_model_start (NautilusSearchProvider *provider)
{
NautilusSearchEngineModel *model;
model = NAUTILUS_SEARCH_ENGINE_MODEL (provider);
if (model->query_pending)
{
return;
}
DEBUG ("Model engine start");
g_object_ref (model);
model->query_pending = TRUE;
g_object_notify (G_OBJECT (provider), "running");
if (model->directory == NULL)
{
search_finished_idle (model);
return;
}
nautilus_directory_call_when_ready (model->directory,
NAUTILUS_FILE_ATTRIBUTE_INFO,
TRUE, model_directory_ready_cb, model);
}
static void
nautilus_search_engine_model_stop (NautilusSearchProvider *provider)
{
NautilusSearchEngineModel *model;
model = NAUTILUS_SEARCH_ENGINE_MODEL (provider);
if (model->query_pending)
{
DEBUG ("Model engine stop");
nautilus_directory_cancel_callback (model->directory,
model_directory_ready_cb, model);
search_finished_idle (model);
}
g_clear_object (&model->directory);
}
static void
nautilus_search_engine_model_set_query (NautilusSearchProvider *provider,
NautilusQuery *query)
{
NautilusSearchEngineModel *model;
model = NAUTILUS_SEARCH_ENGINE_MODEL (provider);
g_object_ref (query);
g_clear_object (&model->query);
model->query = query;
}
static gboolean
nautilus_search_engine_model_is_running (NautilusSearchProvider *provider)
{
NautilusSearchEngineModel *model;
model = NAUTILUS_SEARCH_ENGINE_MODEL (provider);
return model->query_pending;
}
static void
nautilus_search_provider_init (NautilusSearchProviderInterface *iface)
{
iface->set_query = nautilus_search_engine_model_set_query;
iface->start = nautilus_search_engine_model_start;
iface->stop = nautilus_search_engine_model_stop;
iface->is_running = nautilus_search_engine_model_is_running;
}
static void
nautilus_search_engine_model_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
NautilusSearchProvider *self = NAUTILUS_SEARCH_PROVIDER (object);
switch (prop_id)
{
case PROP_RUNNING:
{
g_value_set_boolean (value, nautilus_search_engine_model_is_running (self));
}
break;
default:
{
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
}
static void
nautilus_search_engine_model_class_init (NautilusSearchEngineModelClass *class)
{
GObjectClass *gobject_class;
gobject_class = G_OBJECT_CLASS (class);
gobject_class->finalize = finalize;
gobject_class->get_property = nautilus_search_engine_model_get_property;
/**
* NautilusSearchEngine::running:
*
* Whether the search engine is running a search.
*/
g_object_class_override_property (gobject_class, PROP_RUNNING, "running");
}
static void
nautilus_search_engine_model_init (NautilusSearchEngineModel *engine)
{
}
NautilusSearchEngineModel *
nautilus_search_engine_model_new (void)
{
NautilusSearchEngineModel *engine;
engine = g_object_new (NAUTILUS_TYPE_SEARCH_ENGINE_MODEL, NULL);
return engine;
}
void
nautilus_search_engine_model_set_model (NautilusSearchEngineModel *model,
NautilusDirectory *directory)
{
g_clear_object (&model->directory);
model->directory = nautilus_directory_ref (directory);
}
NautilusDirectory *
nautilus_search_engine_model_get_model (NautilusSearchEngineModel *model)
{
return model->directory;
}