forked from AuroraMiddleware/gtk
b890e87818
The filechooser tries to figure out whether it got results by poking the model, but all files might go through the async GFileInfo querying state. Make all search engines (and the composite one) just notify about this fact, so the file chooser can behave appropriately without waiting for the async operations to finish.
271 lines
6.7 KiB
C
271 lines
6.7 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 <quartz/gdkquartz.h>
|
|
|
|
#include "gtksearchenginequartz.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 the Beagle backend. */
|
|
const unsigned int max_hits = 1000;
|
|
const unsigned int max_iter = submitted_hits + [ns_query resultCount];
|
|
|
|
/* Here we submit hits "submitted_hits" to "resultCount" */
|
|
for (i = submitted_hits; 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];
|
|
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;
|
|
|
|
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;
|
|
|
|
/* 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?).
|
|
*/
|
|
[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_QUARTZ
|
|
return g_object_new (GTK_TYPE_SEARCH_ENGINE_QUARTZ, NULL);
|
|
#else
|
|
return NULL;
|
|
#endif
|
|
}
|