2008-02-11 07:45:34 +00:00
|
|
|
//
|
|
|
|
// SpotlightWindowController.m
|
|
|
|
// Cog
|
|
|
|
//
|
|
|
|
// Created by Matthew Grinshpun on 10/02/08.
|
2008-02-14 14:07:10 +00:00
|
|
|
// Copyright 2008 Matthew Leon Grinshpun. All rights reserved.
|
2008-02-11 07:45:34 +00:00
|
|
|
//
|
|
|
|
|
|
|
|
#import "SpotlightWindowController.h"
|
2008-02-16 13:05:30 +00:00
|
|
|
#import "NSArray+CogSort.h"
|
2022-02-07 05:49:27 +00:00
|
|
|
#import "NSComparisonPredicate+CogPredicate.h"
|
2008-02-16 13:18:14 +00:00
|
|
|
#import "NSNumber+CogSort.h"
|
2022-02-07 05:49:27 +00:00
|
|
|
#import "NSString+CogSort.h"
|
|
|
|
#import "PlaylistLoader.h"
|
|
|
|
#import "SpotlightPlaylistEntry.h"
|
2008-02-16 22:59:27 +00:00
|
|
|
#import "SpotlightTransformers.h"
|
2008-02-11 07:45:34 +00:00
|
|
|
|
2013-10-11 12:03:55 +00:00
|
|
|
#import "Logging.h"
|
|
|
|
|
2008-02-13 23:51:36 +00:00
|
|
|
// Minimum length of a search string (searching for very small strings gets ugly)
|
|
|
|
#define MINIMUM_SEARCH_STRING_LENGTH 3
|
|
|
|
|
|
|
|
// Store a class predicate for searching for music
|
2022-02-07 05:49:27 +00:00
|
|
|
static NSPredicate *musicOnlyPredicate = nil;
|
2008-02-11 07:45:34 +00:00
|
|
|
|
|
|
|
@implementation SpotlightWindowController
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
+ (void)initialize {
|
2016-05-05 20:05:39 +00:00
|
|
|
musicOnlyPredicate = [NSPredicate predicateWithFormat:
|
2022-02-07 05:49:27 +00:00
|
|
|
@"kMDItemContentTypeTree==\'public.audio\'"];
|
|
|
|
|
|
|
|
// Register value transformers
|
|
|
|
NSValueTransformer *pausingQueryTransformer = [[PausingQueryTransformer alloc] init];
|
|
|
|
[NSValueTransformer setValueTransformer:pausingQueryTransformer forName:@"PausingQueryTransformer"];
|
|
|
|
|
|
|
|
NSValueTransformer *authorToArtistTransformer = [[AuthorToArtistTransformer alloc] init];
|
|
|
|
[NSValueTransformer setValueTransformer:authorToArtistTransformer forName:@"AuthorToArtistTransformer"];
|
|
|
|
|
|
|
|
NSValueTransformer *pathToURLTransformer = [[PathToURLTransformer alloc] init];
|
2022-07-15 10:02:41 +00:00
|
|
|
[NSValueTransformer setValueTransformer:pathToURLTransformer forName:@"PathToURLTransformer"];
|
2022-02-07 05:49:27 +00:00
|
|
|
|
|
|
|
NSValueTransformer *stringToSearchScopeTransformer = [[StringToSearchScopeTransformer alloc] init];
|
|
|
|
[NSValueTransformer setValueTransformer:stringToSearchScopeTransformer forName:@"StringToSearchScopeTransformer"];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)registerDefaults {
|
|
|
|
// Set the home directory as the default search directory
|
|
|
|
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
|
|
|
|
NSString *homeDir = @"~";
|
|
|
|
homeDir = [homeDir stringByExpandingTildeInPath];
|
|
|
|
homeDir = [[NSURL fileURLWithPath:homeDir isDirectory:YES] absoluteString];
|
2022-02-09 03:42:03 +00:00
|
|
|
NSDictionary *searchDefault = @{@"spotlightSearchPath": homeDir};
|
2022-02-07 05:49:27 +00:00
|
|
|
[defaults registerDefaults:searchDefault];
|
2008-02-23 14:09:34 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (id)init {
|
|
|
|
if(self = [super initWithWindowNibName:@"SpotlightPanel"]) {
|
|
|
|
self.query = [[NSMetadataQuery alloc] init];
|
|
|
|
[self.query setDelegate:self];
|
|
|
|
self.query.sortDescriptors = @[
|
|
|
|
[[NSSortDescriptor alloc] initWithKey:@"kMDItemAuthors"
|
|
|
|
ascending:YES
|
|
|
|
selector:@selector(compareFirstString:)],
|
|
|
|
[[NSSortDescriptor alloc] initWithKey:@"kMDItemAlbum"
|
|
|
|
ascending:YES
|
|
|
|
selector:@selector(caseInsensitiveCompare:)],
|
|
|
|
[[NSSortDescriptor alloc] initWithKey:@"kMDItemAudioTrackNumber"
|
|
|
|
ascending:YES
|
|
|
|
selector:@selector(compareTrackNumbers:)]
|
|
|
|
];
|
|
|
|
|
|
|
|
// hook my query transformer up to me
|
|
|
|
[PausingQueryTransformer setSearchController:self];
|
2008-03-13 02:07:49 +00:00
|
|
|
|
|
|
|
[self registerDefaults];
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
// TODO: spotlightSearchPath is bound via IB, is the below needed?
|
|
|
|
// NSDictionary *bindOptions =
|
|
|
|
// [NSDictionary dictionaryWithObject:@"StringToSearchScopeTransformer"
|
|
|
|
// forKey:NSValueTransformerNameBindingOption];
|
|
|
|
//
|
|
|
|
// [self.query bind:@"searchScopes"
|
|
|
|
// toObject:[NSUserDefaultsController sharedUserDefaultsController]
|
|
|
|
// withKeyPath:@"values.spotlightSearchPath"
|
|
|
|
// options:bindOptions];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
return self;
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)awakeFromNib {
|
2008-02-18 20:46:53 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (IBAction)toggleWindow:(id)sender {
|
|
|
|
if([[self window] isVisible])
|
2008-02-21 10:45:09 +00:00
|
|
|
[[self window] orderOut:self];
|
|
|
|
else
|
|
|
|
[self showWindow:self];
|
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)performSearch {
|
|
|
|
NSPredicate *searchPredicate;
|
|
|
|
// Process the search string into a compound predicate. If Nil is returned do nothing
|
|
|
|
if((searchPredicate = [self processSearchString])) {
|
|
|
|
// spotlightPredicate, which is what will finally be used for the spotlight search
|
|
|
|
// is the union of the (potentially) compound searchPredicate and the static
|
|
|
|
// musicOnlyPredicate
|
|
|
|
|
|
|
|
NSPredicate *spotlightPredicate = [NSCompoundPredicate andPredicateWithSubpredicates:
|
|
|
|
@[musicOnlyPredicate,
|
|
|
|
searchPredicate]];
|
|
|
|
// Only preform a new search if the predicate has changed or there is a new path
|
|
|
|
if(![self.query.predicate isEqual:spotlightPredicate] || ![self.query.searchScopes isEqualToArray:
|
|
|
|
@[pathControl.URL]]) {
|
|
|
|
if([self.query isStarted])
|
|
|
|
[self.query stopQuery];
|
|
|
|
self.query.predicate = spotlightPredicate;
|
|
|
|
// Set scope to contents of pathControl
|
|
|
|
self.query.searchScopes = @[pathControl.URL];
|
|
|
|
[self.query startQuery];
|
|
|
|
DLog(@"Started query: %@", [self.query.predicate description]);
|
|
|
|
}
|
|
|
|
}
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (NSPredicate *)processSearchString {
|
|
|
|
NSMutableArray *subpredicates = [NSMutableArray arrayWithCapacity:10];
|
|
|
|
|
|
|
|
NSScanner *scanner = [NSScanner scannerWithString:self.searchString];
|
|
|
|
BOOL exactString;
|
|
|
|
NSString *scannedString;
|
|
|
|
NSMutableString *parsingString;
|
|
|
|
while(![scanner isAtEnd]) {
|
|
|
|
exactString = NO;
|
|
|
|
if([scanner scanUpToString:@" " intoString:&scannedString]) {
|
|
|
|
if([scannedString length] < MINIMUM_SEARCH_STRING_LENGTH)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// We use NSMutableString because this string will get abused a bit
|
|
|
|
// It potentially could be reading the entire search string
|
|
|
|
|
|
|
|
parsingString = [NSMutableString stringWithCapacity:[self.searchString length]];
|
|
|
|
[parsingString setString:scannedString];
|
|
|
|
|
|
|
|
if([parsingString characterAtIndex:0] == '%') {
|
|
|
|
if([parsingString length] < (MINIMUM_SEARCH_STRING_LENGTH + 2))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if([parsingString characterAtIndex:2] == '\"') {
|
|
|
|
exactString = YES;
|
|
|
|
// If the string does not end in a quotation mark and we're not at the end,
|
|
|
|
// scan until we find one.
|
|
|
|
// Allows strings within quotation marks to include spaces
|
|
|
|
if([parsingString characterAtIndex:([parsingString length] - 1)] != '\"' &&
|
|
|
|
![scanner isAtEnd]) {
|
|
|
|
NSString *restOfString;
|
|
|
|
[scanner scanUpToString:@"\"" intoString:&restOfString];
|
|
|
|
[parsingString appendFormat:@" %@", restOfString];
|
|
|
|
} else if([parsingString characterAtIndex:([parsingString length] - 1)] == '\"') {
|
|
|
|
// pick off the quotation mark at the end
|
|
|
|
[parsingString deleteCharactersInRange:
|
|
|
|
NSMakeRange([parsingString length] - 1, 1)];
|
|
|
|
}
|
|
|
|
// eliminate beginning quotation mark
|
|
|
|
[parsingString deleteCharactersInRange:NSMakeRange(2, 1)];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for artist
|
|
|
|
if([parsingString characterAtIndex:1] == 'a') {
|
|
|
|
[subpredicates addObject:
|
|
|
|
[NSComparisonPredicate predicateForMdKey:@"kMDItemAuthors"
|
|
|
|
withString:[parsingString substringFromIndex:2]
|
|
|
|
exactString:exactString]];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for album
|
|
|
|
if([parsingString characterAtIndex:1] == 'l') {
|
|
|
|
[subpredicates addObject:
|
|
|
|
[NSComparisonPredicate predicateForMdKey:@"kMDItemAlbum"
|
|
|
|
withString:[parsingString substringFromIndex:2]
|
|
|
|
exactString:exactString]];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for title
|
|
|
|
if([parsingString characterAtIndex:1] == 't') {
|
|
|
|
[subpredicates addObject:
|
|
|
|
[NSComparisonPredicate predicateForMdKey:@"kMDItemTitle"
|
|
|
|
withString:[parsingString substringFromIndex:2]
|
|
|
|
exactString:exactString]];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for genre
|
|
|
|
if([parsingString characterAtIndex:1] == 'g') {
|
|
|
|
[subpredicates addObject:
|
|
|
|
[NSComparisonPredicate predicateForMdKey:@"kMDItemMusicalGenre"
|
|
|
|
withString:[parsingString substringFromIndex:2]
|
|
|
|
exactString:exactString]];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for comment
|
|
|
|
if([parsingString characterAtIndex:1] == 'c') {
|
|
|
|
[subpredicates addObject:
|
|
|
|
[NSComparisonPredicate predicateForMdKey:@"kMDItemComment"
|
|
|
|
withString:[parsingString substringFromIndex:2]
|
|
|
|
exactString:exactString]];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
NSString *wildcardString = [NSString stringWithFormat:@"*%@*", parsingString];
|
|
|
|
NSPredicate *pred = [NSPredicate predicateWithFormat:@"(kMDItemTitle LIKE[cd] %@) OR (kMDItemAlbum LIKE[cd] %@) OR (kMDItemAuthors LIKE[cd] %@)",
|
|
|
|
wildcardString, wildcardString, wildcardString];
|
|
|
|
[subpredicates addObject:pred];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if([subpredicates count] == 0)
|
|
|
|
return nil;
|
|
|
|
else if([subpredicates count] == 1)
|
|
|
|
return [subpredicates objectAtIndex:0];
|
|
|
|
|
|
|
|
// Create a compound predicate from subPredicates
|
|
|
|
return [NSCompoundPredicate andPredicateWithSubpredicates:subpredicates];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)searchForArtist:(NSString *)artist {
|
|
|
|
[self showWindow:self];
|
|
|
|
self.searchString = [NSString stringWithFormat:@"%%a\"%@\"", artist];
|
2008-02-16 20:08:45 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)searchForAlbum:(NSString *)album {
|
|
|
|
[self showWindow:self];
|
|
|
|
self.searchString = [NSString stringWithFormat:@"%%l\"%@\"", album];
|
2008-02-16 20:08:45 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)dealloc {
|
2008-02-18 20:09:02 +00:00
|
|
|
self.query = nil;
|
|
|
|
self.searchString = nil;
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-07 05:49:27 +00:00
|
|
|
- (IBAction)addToPlaylist:(id)sender {
|
|
|
|
NSArray *tracks;
|
|
|
|
[self.query disableUpdates];
|
|
|
|
tracks = playlistController.selectedObjects;
|
|
|
|
if([tracks count] == 0)
|
|
|
|
tracks = playlistController.arrangedObjects;
|
|
|
|
|
2022-07-06 21:28:14 +00:00
|
|
|
NSDictionary *loadEntryData = @{ @"entries": [tracks valueForKey:@"url"],
|
|
|
|
@"sort": @(NO),
|
|
|
|
@"origin": @(URLOriginExternal) };
|
|
|
|
|
|
|
|
[playlistLoader performSelectorInBackground:@selector(addURLsInBackground:) withObject:loadEntryData];
|
2008-05-09 21:24:49 +00:00
|
|
|
[self.query enableUpdates];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
2013-10-11 13:35:53 +00:00
|
|
|
// If pop-up styled NSPathControl is set to /a/b/c path, then selecting either 'a' or 'b'
|
|
|
|
// from its pop-up menu won't do anything by default (while we'd like it to select /a and
|
|
|
|
// /a/b respectively). So here we set url of NSPathControl to be that of clicked cell.
|
2022-02-07 05:49:27 +00:00
|
|
|
- (IBAction)pathComponentClicked:(id)sender {
|
|
|
|
NSPathComponentCell *pcc = [sender clickedPathComponentCell];
|
|
|
|
DLog(@"%@", pcc);
|
|
|
|
[sender setURL:[pcc URL]];
|
2013-10-11 13:35:53 +00:00
|
|
|
}
|
|
|
|
|
2008-02-13 23:51:36 +00:00
|
|
|
#pragma mark NSMetadataQuery delegate methods
|
|
|
|
|
|
|
|
// replace the NSMetadataItem with a PlaylistEntry
|
2022-02-07 05:49:27 +00:00
|
|
|
- (id)metadataQuery:(NSMetadataQuery *)query
|
|
|
|
replacementObjectForResultObject:(NSMetadataItem *)result {
|
|
|
|
return [SpotlightPlaylistEntry playlistEntryWithMetadataItem:result];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark Getters and setters
|
|
|
|
|
|
|
|
@synthesize query;
|
|
|
|
|
|
|
|
@synthesize searchString;
|
2022-02-07 05:49:27 +00:00
|
|
|
- (void)setSearchString:(NSString *)aString {
|
2008-02-13 23:51:36 +00:00
|
|
|
// Make sure the string is changed
|
2022-02-07 05:49:27 +00:00
|
|
|
if(![searchString isEqualToString:aString]) {
|
2008-02-13 23:51:36 +00:00
|
|
|
searchString = [aString copy];
|
2022-02-07 05:49:27 +00:00
|
|
|
[self performSearch];
|
2008-02-13 23:51:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-11 07:45:34 +00:00
|
|
|
@end
|