296 lines
7.4 KiB
C
296 lines
7.4 KiB
C
|
/*
|
||
|
* Copyright (C) 2007 Kristian Rietveld <kris@gtk.org>
|
||
|
*
|
||
|
* 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/>.
|
||
|
*/
|
||
|
|
||
|
#include <Cocoa/Cocoa.h>
|
||
|
#include <macos/gdkmacos.h>
|
||
|
|
||
|
#include "gtksearchenginequartzprivate.h"
|
||
|
|
||
|
/* This file is a mixture of an objective-C object and a GObject,
|
||
|
* so be careful to not confuse yourself.
|
||
|
*/
|
||
|
|
||
|
#define QUARTZ_POOL_ALLOC NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]
|
||
|
#define QUARTZ_POOL_RELEASE [pool release]
|
||
|
|
||
|
|
||
|
/* Definition of objective-c object */
|
||
|
@interface ResultReceiver : NSObject
|
||
|
{
|
||
|
int submitted_hits;
|
||
|
GtkSearchEngine *engine;
|
||
|
}
|
||
|
|
||
|
- (void) setEngine:(GtkSearchEngine *)quartz_engine;
|
||
|
|
||
|
- (void) queryUpdate:(id)sender;
|
||
|
- (void) queryProgress:(id)sender;
|
||
|
- (void) queryFinished:(id)sender;
|
||
|
|
||
|
@end
|
||
|
|
||
|
|
||
|
/* Definition of GObject */
|
||
|
struct _GtkSearchEngineQuartzPrivate
|
||
|
{
|
||
|
GtkQuery *query;
|
||
|
|
||
|
ResultReceiver *receiver;
|
||
|
NSMetadataQuery *ns_query;
|
||
|
|
||
|
gboolean query_finished;
|
||
|
};
|
||
|
|
||
|
G_DEFINE_TYPE_WITH_PRIVATE (GtkSearchEngineQuartz, _gtk_search_engine_quartz, GTK_TYPE_SEARCH_ENGINE)
|
||
|
|
||
|
|
||
|
/* Implementation of the objective-C object */
|
||
|
@implementation ResultReceiver
|
||
|
|
||
|
- (void) setEngine:(GtkSearchEngine *)quartz_engine
|
||
|
{
|
||
|
g_return_if_fail (GTK_IS_SEARCH_ENGINE (quartz_engine));
|
||
|
|
||
|
engine = quartz_engine;
|
||
|
submitted_hits = 0;
|
||
|
}
|
||
|
|
||
|
- (void) submitHits:(NSMetadataQuery *)ns_query
|
||
|
{
|
||
|
int i;
|
||
|
GList *hits = NULL;
|
||
|
/* The max was originally set to 1000 to mimic something called "the
|
||
|
* boogie backend". submitted_hits contains the number of hits we've
|
||
|
* processed in previous calls to this function.
|
||
|
*/
|
||
|
const unsigned int max_hits = 1000 - submitted_hits;
|
||
|
const unsigned int max_iter = [ns_query resultCount];
|
||
|
|
||
|
for (i = 0; i < max_iter && i < max_hits; ++i)
|
||
|
{
|
||
|
id result = [ns_query resultAtIndex:i];
|
||
|
const char *result_path;
|
||
|
GFile *file;
|
||
|
GtkSearchHit *hit;
|
||
|
|
||
|
result_path = [[result valueForAttribute:@"kMDItemPath"] UTF8String];
|
||
|
|
||
|
if (result_path == NULL)
|
||
|
continue;
|
||
|
|
||
|
file = g_file_new_for_path (result_path);
|
||
|
|
||
|
hit = g_new (GtkSearchHit, 1);
|
||
|
hit->file = file;
|
||
|
hit->info = NULL;
|
||
|
|
||
|
hits = g_list_prepend (hits, hit);
|
||
|
}
|
||
|
|
||
|
_gtk_search_engine_hits_added (engine, hits);
|
||
|
g_list_free_full (hits, (GDestroyNotify) _gtk_search_hit_free);
|
||
|
|
||
|
if (max_iter >= max_hits)
|
||
|
[ns_query stopQuery];
|
||
|
|
||
|
submitted_hits += max_iter;
|
||
|
}
|
||
|
|
||
|
- (void) queryUpdate:(id)sender
|
||
|
{
|
||
|
NSMetadataQuery *ns_query = [sender object];
|
||
|
|
||
|
[self submitHits:ns_query];
|
||
|
}
|
||
|
|
||
|
- (void) queryProgress:(id)sender
|
||
|
{
|
||
|
NSMetadataQuery *ns_query = [sender object];
|
||
|
|
||
|
[self submitHits:ns_query];
|
||
|
}
|
||
|
|
||
|
- (void) queryFinished:(id)sender
|
||
|
{
|
||
|
NSMetadataQuery *ns_query = [sender object];
|
||
|
|
||
|
[self submitHits:ns_query];
|
||
|
|
||
|
_gtk_search_engine_finished (engine, submitted_hits > 0);
|
||
|
submitted_hits = 0;
|
||
|
}
|
||
|
|
||
|
@end
|
||
|
|
||
|
/* Implementation of the GObject */
|
||
|
|
||
|
static void
|
||
|
gtk_search_engine_quartz_finalize (GObject *object)
|
||
|
{
|
||
|
GtkSearchEngineQuartz *quartz;
|
||
|
|
||
|
QUARTZ_POOL_ALLOC;
|
||
|
|
||
|
quartz = GTK_SEARCH_ENGINE_QUARTZ (object);
|
||
|
|
||
|
[[NSNotificationCenter defaultCenter] removeObserver:quartz->priv->receiver];
|
||
|
|
||
|
[quartz->priv->ns_query release];
|
||
|
[quartz->priv->receiver release];
|
||
|
|
||
|
QUARTZ_POOL_RELEASE;
|
||
|
|
||
|
if (quartz->priv->query)
|
||
|
{
|
||
|
g_object_unref (quartz->priv->query);
|
||
|
quartz->priv->query = NULL;
|
||
|
}
|
||
|
|
||
|
G_OBJECT_CLASS (_gtk_search_engine_quartz_parent_class)->finalize (object);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gtk_search_engine_quartz_start (GtkSearchEngine *engine)
|
||
|
{
|
||
|
GtkSearchEngineQuartz *quartz;
|
||
|
|
||
|
QUARTZ_POOL_ALLOC;
|
||
|
|
||
|
quartz = GTK_SEARCH_ENGINE_QUARTZ (engine);
|
||
|
|
||
|
[quartz->priv->ns_query startQuery];
|
||
|
|
||
|
QUARTZ_POOL_RELEASE;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gtk_search_engine_quartz_stop (GtkSearchEngine *engine)
|
||
|
{
|
||
|
GtkSearchEngineQuartz *quartz;
|
||
|
|
||
|
QUARTZ_POOL_ALLOC;
|
||
|
|
||
|
quartz = GTK_SEARCH_ENGINE_QUARTZ (engine);
|
||
|
|
||
|
[quartz->priv->ns_query stopQuery];
|
||
|
|
||
|
QUARTZ_POOL_RELEASE;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gtk_search_engine_quartz_set_query (GtkSearchEngine *engine,
|
||
|
GtkQuery *query)
|
||
|
{
|
||
|
GtkSearchEngineQuartz *quartz;
|
||
|
const char* path = NULL;
|
||
|
GFile *location = NULL;
|
||
|
|
||
|
QUARTZ_POOL_ALLOC;
|
||
|
|
||
|
quartz = GTK_SEARCH_ENGINE_QUARTZ (engine);
|
||
|
|
||
|
if (query)
|
||
|
g_object_ref (query);
|
||
|
|
||
|
if (quartz->priv->query)
|
||
|
g_object_unref (quartz->priv->query);
|
||
|
|
||
|
quartz->priv->query = query;
|
||
|
location = gtk_query_get_location (query);
|
||
|
|
||
|
if (location)
|
||
|
path = g_file_peek_path (location);
|
||
|
|
||
|
/* We create a query to look for ".*text.*" in the text contents of
|
||
|
* all indexed files. (Should we also search for text in file and folder
|
||
|
* names?).
|
||
|
*/
|
||
|
|
||
|
if (path)
|
||
|
{
|
||
|
NSString *ns_path = [[NSString string] initWithUTF8String:path];
|
||
|
[quartz->priv->ns_query setSearchScopes:@[ns_path]];
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
[quartz->priv->ns_query setSearchScopes:@[NSMetadataQueryLocalComputerScope]];
|
||
|
}
|
||
|
|
||
|
[quartz->priv->ns_query setSearchItems:@[(NSString*)kMDItemTextContent,
|
||
|
(NSString*)kMDItemFSName]];
|
||
|
[quartz->priv->ns_query setPredicate:
|
||
|
[NSPredicate predicateWithFormat:
|
||
|
[NSString stringWithFormat:@"(kMDItemTextContent LIKE[cd] \"*%s*\")",
|
||
|
gtk_query_get_text (query)]]];
|
||
|
|
||
|
QUARTZ_POOL_RELEASE;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
_gtk_search_engine_quartz_class_init (GtkSearchEngineQuartzClass *class)
|
||
|
{
|
||
|
GObjectClass *gobject_class;
|
||
|
GtkSearchEngineClass *engine_class;
|
||
|
|
||
|
gobject_class = G_OBJECT_CLASS (class);
|
||
|
gobject_class->finalize = gtk_search_engine_quartz_finalize;
|
||
|
|
||
|
engine_class = GTK_SEARCH_ENGINE_CLASS (class);
|
||
|
engine_class->set_query = gtk_search_engine_quartz_set_query;
|
||
|
engine_class->start = gtk_search_engine_quartz_start;
|
||
|
engine_class->stop = gtk_search_engine_quartz_stop;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
_gtk_search_engine_quartz_init (GtkSearchEngineQuartz *engine)
|
||
|
{
|
||
|
QUARTZ_POOL_ALLOC;
|
||
|
|
||
|
engine->priv = _gtk_search_engine_quartz_get_instance_private (engine);
|
||
|
|
||
|
engine->priv->ns_query = [[NSMetadataQuery alloc] init];
|
||
|
engine->priv->receiver = [[ResultReceiver alloc] init];
|
||
|
|
||
|
[engine->priv->receiver setEngine:GTK_SEARCH_ENGINE (engine)];
|
||
|
|
||
|
[[NSNotificationCenter defaultCenter] addObserver:engine->priv->receiver
|
||
|
selector:@selector(queryUpdate:)
|
||
|
name:@"NSMetadataQueryDidUpdateNotification"
|
||
|
object:engine->priv->ns_query];
|
||
|
[[NSNotificationCenter defaultCenter] addObserver:engine->priv->receiver
|
||
|
selector:@selector(queryFinished:)
|
||
|
name:@"NSMetadataQueryDidFinishGatheringNotification"
|
||
|
object:engine->priv->ns_query];
|
||
|
|
||
|
[[NSNotificationCenter defaultCenter] addObserver:engine->priv->receiver
|
||
|
selector:@selector(queryProgress:)
|
||
|
name:@"NSMetadataQueryGatheringProgressNotification"
|
||
|
object:engine->priv->ns_query];
|
||
|
|
||
|
QUARTZ_POOL_RELEASE;
|
||
|
}
|
||
|
|
||
|
GtkSearchEngine *
|
||
|
_gtk_search_engine_quartz_new (void)
|
||
|
{
|
||
|
#ifdef GDK_WINDOWING_MACOS
|
||
|
return g_object_new (GTK_TYPE_SEARCH_ENGINE_QUARTZ, NULL);
|
||
|
#else
|
||
|
return NULL;
|
||
|
#endif
|
||
|
}
|