/* Copyright 2010 Hardcoded Software (http://www.hardcoded.net) This software is licensed under the "HS" License as described in the "LICENSE" file, which should be included with this package. The terms are also available at http://www.hardcoded.net/licenses/hs_license */ #import "ResultWindow.h" #import "Dialogs.h" #import "ProgressController.h" #import "Utils.h" #import "RegistrationInterface.h" #import "AppDelegate.h" #import "Consts.h" @implementation MatchesView - (void)keyDown:(NSEvent *)theEvent { unichar key = [[theEvent charactersIgnoringModifiers] characterAtIndex:0]; // get flags and strip the lower 16 (device dependant) bits NSUInteger flags = ( [theEvent modifierFlags] & 0x00FF ); if (((key == NSDeleteFunctionKey) || (key == NSDeleteCharacter)) && (flags == 0)) [self sendAction:@selector(removeSelected:) to:[self delegate]]; else if ((key == 0x20) && (flags == 0)) // Space [self sendAction:@selector(markSelected:) to:[self delegate]]; else [super keyDown:theEvent]; } - (void)outlineView:(NSOutlineView *)outlineView setObjectValue:(id)object forTableColumn:(NSTableColumn *)tableColumn byItem:(id)item { if (![[tableColumn identifier] isEqual:@"0"]) return; //We only want to cover renames. OVNode *node = item; NSString *oldName = [[node buffer] objectAtIndex:0]; NSString *newName = object; if (![newName isEqual:oldName]) { BOOL renamed = n2b([(PyDupeGuruBase *)py renameSelected:newName]); if (renamed) [[NSNotificationCenter defaultCenter] postNotificationName:ResultsChangedNotification object:self]; else [Dialogs showMessage:[NSString stringWithFormat:@"The name '%@' already exists.",newName]]; } } @end @implementation ResultWindowBase - (void)awakeFromNib { [self window]; preferencesPanel = [[NSWindowController alloc] initWithWindowNibName:@"Preferences"]; [self initResultColumns]; [self fillColumnsMenu]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(registrationRequired:) name:RegistrationRequired object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(jobCompleted:) name:JobCompletedNotification object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(jobStarted:) name:JobStarted object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(jobInProgress:) name:JobInProgress object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(resultsChanged:) name:ResultsChangedNotification object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(resultsUpdated:) name:ResultsUpdatedNotification object:nil]; } - (void)dealloc { [preferencesPanel release]; [super dealloc]; } /* Helpers */ - (void)fillColumnsMenu { // The columns menu is supposed to be empty and initResultColumns must have been called for (NSTableColumn *col in _resultColumns) { NSMenuItem *mi = [columnsMenu addItemWithTitle:[[col headerCell] stringValue] action:@selector(toggleColumn:) keyEquivalent:@""]; [mi setTag:[[col identifier] integerValue]]; [mi setTarget:self]; if ([[matches tableColumns] containsObject:col]) [mi setState:NSOnState]; } [columnsMenu addItem:[NSMenuItem separatorItem]]; NSMenuItem *mi = [columnsMenu addItemWithTitle:@"Reset to Default" action:@selector(resetColumnsToDefault:) keyEquivalent:@""]; [mi setTarget:self]; } - (NSTableColumn *)getColumnForIdentifier:(NSInteger)aIdentifier title:(NSString *)aTitle width:(NSInteger)aWidth refCol:(NSTableColumn *)aColumn { NSNumber *n = [NSNumber numberWithInteger:aIdentifier]; NSTableColumn *col = [[NSTableColumn alloc] initWithIdentifier:[n stringValue]]; [col setWidth:aWidth]; [col setEditable:NO]; [[col dataCell] setFont:[[aColumn dataCell] font]]; [[col headerCell] setStringValue:aTitle]; [col setResizingMask:NSTableColumnUserResizingMask]; [col setSortDescriptorPrototype:[[NSSortDescriptor alloc] initWithKey:[n stringValue] ascending:YES]]; return col; } //Returns an array of identifiers, in order. - (NSArray *)getColumnsOrder { NSTableColumn *col; NSString *colId; NSMutableArray *result = [NSMutableArray array]; NSEnumerator *e = [[matches tableColumns] objectEnumerator]; while (col = [e nextObject]) { colId = [col identifier]; [result addObject:colId]; } return result; } - (NSDictionary *)getColumnsWidth { NSMutableDictionary *result = [NSMutableDictionary dictionary]; NSTableColumn *col; NSString *colId; NSNumber *width; NSEnumerator *e = [[matches tableColumns] objectEnumerator]; while (col = [e nextObject]) { colId = [col identifier]; width = [NSNumber numberWithDouble:[col width]]; [result setObject:width forKey:colId]; } return result; } - (NSArray *)getSelected:(BOOL)aDupesOnly { if (_powerMode) aDupesOnly = NO; NSIndexSet *indexes = [matches selectedRowIndexes]; NSMutableArray *nodeList = [NSMutableArray array]; OVNode *node; NSInteger i = [indexes firstIndex]; while (i != NSNotFound) { node = [matches itemAtRow:i]; if (!aDupesOnly || ([node level] > 1)) [nodeList addObject:node]; i = [indexes indexGreaterThanIndex:i]; } return nodeList; } - (NSArray *)getSelectedPaths:(BOOL)aDupesOnly { NSMutableArray *r = [NSMutableArray array]; NSArray *selected = [self getSelected:aDupesOnly]; NSEnumerator *e = [selected objectEnumerator]; OVNode *node; while (node = [e nextObject]) [r addObject:p2a([node indexPath])]; return r; } - (void)initResultColumns { // Virtual } - (void)restoreColumnsPosition:(NSArray *)aColumnsOrder widths:(NSDictionary *)aColumnsWidth { NSTableColumn *col; NSString *colId; NSNumber *width; NSMenuItem *mi; //Remove all columns NSEnumerator *e = [[columnsMenu itemArray] objectEnumerator]; while (mi = [e nextObject]) { if ([mi state] == NSOnState) [self toggleColumn:mi]; } //Add columns and set widths e = [aColumnsOrder objectEnumerator]; while (colId = [e nextObject]) { if (![colId isEqual:@"mark"]) { col = [_resultColumns objectAtIndex:[colId intValue]]; width = [aColumnsWidth objectForKey:[col identifier]]; mi = [columnsMenu itemWithTag:[colId intValue]]; if (width) [col setWidth:[width floatValue]]; [self toggleColumn:mi]; } } } - (void)updatePySelection { NSArray *selection; if (_powerMode) selection = [py selectedPowerMarkerNodePaths]; else selection = [py selectedResultNodePaths]; [matches selectNodePaths:selection]; } - (void)performPySelection:(NSArray *)aIndexPaths { if (_powerMode) [py selectPowerMarkerNodePaths:aIndexPaths]; else [py selectResultNodePaths:aIndexPaths]; } - (void)refreshStats { [stats setStringValue:[py getStatLine]]; } /* Actions */ - (IBAction)changeDelta:(id)sender { _displayDelta = [deltaSwitch selectedSegment] == 1; [py setDisplayDeltaValues:b2n(_displayDelta)]; [matches reloadData]; [self expandAll:nil]; } - (IBAction)changePowerMarker:(id)sender { _powerMode = [pmSwitch selectedSegment] == 1; if (_powerMode) [matches setTag:2]; else [matches setTag:0]; [self expandAll:nil]; [self outlineView:matches didClickTableColumn:nil]; [self updatePySelection]; } - (IBAction)copyMarked:(id)sender { NSInteger mark_count = [[py getMarkCount] intValue]; if (!mark_count) return; NSOpenPanel *op = [NSOpenPanel openPanel]; [op setCanChooseFiles:NO]; [op setCanChooseDirectories:YES]; [op setCanCreateDirectories:YES]; [op setAllowsMultipleSelection:NO]; [op setTitle:@"Select a directory to copy marked files to"]; if ([op runModalForTypes:nil] == NSOKButton) { NSString *directory = [[op filenames] objectAtIndex:0]; NSUserDefaults *ud = [NSUserDefaults standardUserDefaults]; [py copyOrMove:b2n(YES) markedTo:directory recreatePath:[ud objectForKey:@"recreatePathType"]]; } } - (IBAction)deleteMarked:(id)sender { NSInteger mark_count = [[py getMarkCount] intValue]; if (!mark_count) return; if ([Dialogs askYesNo:[NSString stringWithFormat:@"You are about to send %d files to Trash. Continue?",mark_count]] == NSAlertSecondButtonReturn) // NO return; NSUserDefaults *ud = [NSUserDefaults standardUserDefaults]; [py setRemoveEmptyFolders:[ud objectForKey:@"removeEmptyFolders"]]; [py deleteMarked]; } - (IBAction)expandAll:(id)sender { for (NSInteger i=0;i < [matches numberOfRows];i++) [matches expandItem:[matches itemAtRow:i]]; } - (IBAction)exportToXHTML:(id)sender { NSString *exported = [py exportToXHTMLwithColumns:[self getColumnsOrder]]; [[NSWorkspace sharedWorkspace] openFile:exported]; } - (IBAction)moveMarked:(id)sender { NSInteger mark_count = [[py getMarkCount] intValue]; if (!mark_count) return; NSOpenPanel *op = [NSOpenPanel openPanel]; [op setCanChooseFiles:NO]; [op setCanChooseDirectories:YES]; [op setCanCreateDirectories:YES]; [op setAllowsMultipleSelection:NO]; [op setTitle:@"Select a directory to move marked files to"]; if ([op runModalForTypes:nil] == NSOKButton) { NSString *directory = [[op filenames] objectAtIndex:0]; NSUserDefaults *ud = [NSUserDefaults standardUserDefaults]; [py setRemoveEmptyFolders:[ud objectForKey:@"removeEmptyFolders"]]; [py copyOrMove:b2n(NO) markedTo:directory recreatePath:[ud objectForKey:@"recreatePathType"]]; } } - (IBAction)resetColumnsToDefault:(id)sender { // Virtual } - (IBAction)showPreferencesPanel:(id)sender { [preferencesPanel showWindow:sender]; } - (IBAction)switchSelected:(id)sender { // It might look like a complicated way to get the length of the current dupe list on the py side // but after a lot of fussing around, believe it or not, it actually is. NSInteger matchesTag = _powerMode ? 2 : 0; NSInteger startLen = [[py getOutlineView:matchesTag childCountsForPath:[NSArray array]] count]; [self performPySelection:[self getSelectedPaths:YES]]; [py makeSelectedReference]; // In some cases (when in a filtered view in Power Marker mode, it's possible that the demoted // ref is not a part of the filter, making the table smaller. In those cases, we want to do a // complete reload of the table to avoid a crash. if ([[py getOutlineView:matchesTag childCountsForPath:[NSArray array]] count] == startLen) [[NSNotificationCenter defaultCenter] postNotificationName:ResultsUpdatedNotification object:self]; else [[NSNotificationCenter defaultCenter] postNotificationName:ResultsChangedNotification object:self]; } - (IBAction)toggleColumn:(id)sender { NSMenuItem *mi = sender; NSString *colId = [NSString stringWithFormat:@"%d",[mi tag]]; NSTableColumn *col = [matches tableColumnWithIdentifier:colId]; if (col == nil) { //Add Column col = [_resultColumns objectAtIndex:[mi tag]]; [matches addTableColumn:col]; [mi setState:NSOnState]; } else { //Remove column [matches removeTableColumn:col]; [mi setState:NSOffState]; } } - (IBAction)toggleDetailsPanel:(id)sender { [[(AppDelegateBase *)app detailsPanel] toggleVisibility]; } - (IBAction)togglePowerMarker:(id)sender { if ([pmSwitch selectedSegment] == 1) [pmSwitch setSelectedSegment:0]; else [pmSwitch setSelectedSegment:1]; [self changePowerMarker:sender]; } /* Delegate */ - (void)outlineView:(NSOutlineView *)outlineView didClickTableColumn:(NSTableColumn *)tableColumn { if ([[outlineView sortDescriptors] count] < 1) return; NSSortDescriptor *sd = [[outlineView sortDescriptors] objectAtIndex:0]; if (_powerMode) [py sortDupesBy:i2n([[sd key] intValue]) ascending:b2n([sd ascending])]; else [py sortGroupsBy:i2n([[sd key] intValue]) ascending:b2n([sd ascending])]; [matches reloadData]; [self expandAll:nil]; } /* Notifications */ - (void)windowWillClose:(NSNotification *)aNotification { [NSApp hide:NSApp]; } - (void)jobCompleted:(NSNotification *)aNotification { [[NSNotificationCenter defaultCenter] postNotificationName:ResultsChangedNotification object:self]; NSInteger r = n2i([py getOperationalErrorCount]); id lastAction = [[ProgressController mainProgressController] jobId]; if ([lastAction isEqualTo:jobCopy]) { if (r > 0) [Dialogs showMessage:[NSString stringWithFormat:@"%d file(s) couldn't be copied.",r]]; else [Dialogs showMessage:@"All marked files were copied sucessfully."]; } else if ([lastAction isEqualTo:jobMove]) { if (r > 0) [Dialogs showMessage:[NSString stringWithFormat:@"%d file(s) couldn't be moved. They were kept in the results, and still are marked.",r]]; else [Dialogs showMessage:@"All marked files were moved sucessfully."]; } else if ([lastAction isEqualTo:jobDelete]) { if (r > 0) { NSString *msg = @"%d file(s) couldn't be sent to Trash. They were kept in the results, "\ "and still are marked. See the F.A.Q. section in the help file for details."; [Dialogs showMessage:[NSString stringWithFormat:msg,r]]; } else [Dialogs showMessage:@"All marked files were sucessfully sent to Trash."]; } else if ([lastAction isEqualTo:jobScan]) { NSInteger groupCount = [[py getOutlineView:0 childCountsForPath:[NSArray array]] count]; if (groupCount == 0) [Dialogs showMessage:@"No duplicates found."]; } // Re-activate toolbar items right after the progress bar stops showing instead of waiting until // a mouse-over is performed [[[self window] toolbar] validateVisibleItems]; } - (void)jobInProgress:(NSNotification *)aNotification { [Dialogs showMessage:@"A previous action is still hanging in there. You can't start a new one yet. Wait a few seconds, then try again."]; } - (void)jobStarted:(NSNotification *)aNotification { NSDictionary *ui = [aNotification userInfo]; NSString *desc = [ui valueForKey:@"desc"]; [[ProgressController mainProgressController] setJobDesc:desc]; NSString *jobid = [ui valueForKey:@"jobid"]; // NSLog(jobid); [[ProgressController mainProgressController] setJobId:jobid]; [[ProgressController mainProgressController] showSheetForParent:[self window]]; } - (void)registrationRequired:(NSNotification *)aNotification { NSString *msg = @"This is a demo version, which only allows you 10 delete/copy/move actions per session. You cannot continue."; [Dialogs showMessage:msg]; } - (void)outlineViewSelectionDidChange:(NSNotification *)notification { [self performPySelection:[self getSelectedPaths:NO]]; [py refreshDetailsWithSelected]; [[NSNotificationCenter defaultCenter] postNotificationName:DuplicateSelectionChangedNotification object:self]; } - (void)resultsChanged:(NSNotification *)aNotification { [matches reloadData]; [self expandAll:nil]; [self outlineViewSelectionDidChange:nil]; [self refreshStats]; } - (void)resultsMarkingChanged:(NSNotification *)aNotification { [matches invalidateMarkings]; [self refreshStats]; } - (void)resultsUpdated:(NSNotification *)aNotification { [matches invalidateBuffers]; [matches invalidateMarkings]; } - (BOOL)validateToolbarItem:(NSToolbarItem *)theItem { return ![[ProgressController mainProgressController] isShown]; } - (BOOL)validateMenuItem:(NSMenuItem *)item { return ![[ProgressController mainProgressController] isShown]; } @end