gtk2/gtk/gtksearchenginequartz.c
Carlos Garnacho 99031f885e gtksearchengine: Add "got_results" argument to ::finished
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.
2020-05-19 22:03:20 +02:00

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 <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 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_QUARTZ
return g_object_new (GTK_TYPE_SEARCH_ENGINE_QUARTZ, NULL);
#else
return NULL;
#endif
}