// // IconViewController.m // Perforce // // Created by Adam Czubernat on 05.06.2013. // Copyright (c) 2013 Perforce Software, Inc. All rights reserved. // #import "IconViewController.h" #import "PSCollectionView.h" #import "PSCustomScrollView.h" #import "VersionsViewController.h" #import "HistoryViewController.h" @interface IconViewController () { __weak IBOutlet PSCollectionView *collectionView; IBOutlet NSView *loadingOverlay; __weak IBOutlet NSProgressIndicator *loadingOverlayIndicator; NSMutableArray *items; NSOperationQueue *thumbnailsQueue; BOOL isLoading; NSPopover *popover; // QuickLook QLPreviewPanel *quickLookPanel; NSArray *quickLookItems; NSImage *quickLookIconImage; } - (void)setItems:(NSArray *)items; - (void)loadThumbnails; - (void)showLoadingOverlay; @end @implementation IconViewController - (void)loadView { [super loadView]; [collectionView setFocusRingType:NSFocusRingTypeNone]; [collectionView registerForDraggedTypes:@[ NSFilenamesPboardType, P4PasteboardTypeFile ]]; [self setItems:items]; // Watch selection [collectionView addObserver:self forKeyPath:@"selectionIndexes" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionInitial context:NULL]; } - (void)dealloc { [thumbnailsQueue cancelAllOperations]; [collectionView removeObserver:self forKeyPath:@"selectionIndexes"]; } - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { NSArray *selection = [self selectedItems]; // Quicklook if (!quickLookPanel && [QLPreviewPanel sharedPreviewPanelExists]) [[QLPreviewPanel sharedPreviewPanel] updateController]; if (quickLookPanel) { quickLookItems = selection; [quickLookPanel reloadData]; } [super selectionChanged:selection]; } #pragma mark - Private - (void)setItems:(NSArray *)array { items = [array mutableCopy]; [collectionView setContent:items]; [self refresh]; [self loadThumbnails]; } - (void)loadThumbnails { [thumbnailsQueue cancelAllOperations]; thumbnailsQueue = [[NSOperationQueue alloc] init]; thumbnailsQueue.maxConcurrentOperationCount = 3; [items enumerateObjectsUsingBlock:^(P4Item *item, NSUInteger idx, BOOL *stop) { __weak NSCollectionViewItem *itemView = [collectionView itemAtIndex:idx]; // Thumbnail operation NSBlockOperation *op = [[NSBlockOperation alloc] init]; __weak NSOperation *weakOp = op; [op addExecutionBlock:^{ NSImage *image = [item previewWithSize:(NSSize) { 192.0, 160.0 }]; if ([weakOp isCancelled]) return ; [itemView performSelectorOnMainThread:@selector(setImage:) withObject:image waitUntilDone:NO]; }]; [thumbnailsQueue addOperation:op]; }]; } - (void)showLoadingOverlay { [loadingOverlay setFrame:self.view.bounds]; [self.view addSubview:loadingOverlay]; [loadingOverlayIndicator startAnimation:nil]; } #pragma mark - NSCollectionView delegate - (BOOL)collectionView:(NSCollectionView *)aCollectionView writeItemsAtIndexes:(NSIndexSet *)indexes toPasteboard:(NSPasteboard *)pasteboard { NSArray *draggedItems = [items objectsAtIndexes:indexes]; return [self writeItems:draggedItems toPasteboard:pasteboard]; } - (NSArray *)collectionView:(NSCollectionView *)collectionView namesOfPromisedFilesDroppedAtDestination:(NSURL *)dropURL forDraggedItemsAtIndexes:(NSIndexSet *)indexes { NSArray *draggedItems = [items objectsAtIndexes:indexes]; return [self namesOfPromisedItems:draggedItems droppedAtDestination:dropURL]; } - (NSImage *)collectionView:(NSCollectionView *)aCollectionView draggingImageForItemsAtIndexes:(NSIndexSet *)indexes withEvent:(NSEvent *)event offset:(NSPointPointer)dragImageOffset { NSImage *result = [collectionView draggingImageForItemsAtIndexes:indexes withEvent:event offset:dragImageOffset]; [result lockFocus]; if (indexes.count > 1) { NSPoint mouse = (CGPoint) { result.size.width * 0.5f - dragImageOffset->x, result.size.height * 0.5f - dragImageOffset->y, }; NSShadow *shadow = [[NSShadow alloc] init]; [shadow setShadowOffset:NSMakeSize(0.5, 0.5)]; [shadow setShadowBlurRadius:5.0]; [shadow setShadowColor:[NSColor blackColor]]; NSDictionary *attrs = @{ NSShadowAttributeName : shadow, NSForegroundColorAttributeName : [NSColor whiteColor], }; NSInteger cornerSize = 10.0f; NSGradient *gradient = [[NSGradient alloc] initWithStartingColor:[NSColor colorWithHexString:@"#FCC"] endingColor:[NSColor redColor]]; NSBezierPath *bezier = [NSBezierPath bezierPathWithRoundedRect:(CGRect) { mouse, {20.0f, 20.0f} } xRadius:cornerSize yRadius:cornerSize]; [gradient drawInBezierPath:bezier angle:90.0f]; BOOL copy = ([NSApp currentEvent].modifierFlags & NSAlternateKeyMask) != 0; NSString *str = [NSString stringWithFormat:@"%@%ld", copy ? @"+": @"", indexes.count]; NSAttributedString *attrStr = [[NSAttributedString alloc] initWithString:str attributes:attrs]; [attrStr drawAtPoint:(CGPoint) { mouse.x + (20.0f - attrStr.size.width) * 0.5f, mouse.y + (20.0f - attrStr.size.height) * 0.5f + 1, }]; [result unlockFocus]; } return result; } - (NSDragOperation)collectionView:(NSCollectionView *)aCollectionView validateDrop:(id)draggingInfo proposedIndex:(NSInteger *)proposedDropIndex dropOperation:(NSCollectionViewDropOperation *)proposedDropOperation { // PSLog(@"dropindex %ld operation %d", (long)*proposedDropIndex, (int)*proposedDropOperation); NSDragOperation operation = NSDragOperationEvery; // Check if option key is pressed if ([[NSApp currentEvent] modifierFlags] & NSAlternateKeyMask) operation = NSDragOperationCopy; // Accept only file types if ([[[draggingInfo draggingPasteboard] types] indexOfObject:NSFilenamesPboardType] == -1) return NSDragOperationNone; if (*proposedDropIndex == -1) return NSDragOperationNone; if (*proposedDropOperation == NSCollectionViewDropBefore) { *proposedDropIndex = 0; return operation; } P4Item *target = [items objectAtIndex:*proposedDropIndex]; if (!target.isDirectory || !target.isEditable) { // Can't drop onto a file. Retarget to the column *proposedDropIndex = 0; *proposedDropOperation = NSCollectionViewDropBefore; if (!workingItem.isEditable) return NSDragOperationNone; } return operation; } - (BOOL)collectionView:(NSCollectionView *)aCollectionView acceptDrop:(id)draggingInfo index:(NSInteger)index dropOperation:(NSCollectionViewDropOperation)dropOperation { // Find the target folder P4Item *target = nil; if (dropOperation == NSCollectionViewDropBefore) target = workingItem; else target = [items objectAtIndex:index]; return [self acceptDrop:draggingInfo target:target]; } #pragma mark NSResponder - (void)cancelOperation:(id)sender { NSIndexSet *indexes = [collectionView selectionIndexes]; if ([indexes count] != 1) return; NSInteger index = [indexes lastIndex]; IconViewItem *viewItem = (IconViewItem *)[collectionView itemAtIndex:index]; [viewItem abortEditing]; [self.view.window makeFirstResponder:collectionView]; } - (void)insertNewline:(id)sender { NSIndexSet *indexes = [collectionView selectionIndexes]; if ([indexes count] != 1) return; NSInteger index = [indexes lastIndex]; IconViewItem *viewItem = (IconViewItem *)[collectionView itemAtIndex:index]; P4Item *item = viewItem.representedObject; if (!item.isEditable) return; [viewItem beginEditing]; } #pragma mark - BrowserViewController Overrides - (NSArray *)selectedItems { NSMutableIndexSet *indexes = [[collectionView selectionIndexes] mutableCopy]; // Clip indexes beyond childrens array [indexes removeIndexesInRange:(NSRange) { items.count, NSIntegerMax }]; if (!indexes.count) return nil; return [items objectsAtIndexes:indexes]; } - (void)setWorkingItem:(P4Item *)item { [super setWorkingItem:item]; [collectionView setSelectionIndexes:nil]; [self setItems:workingItem.children]; } - (void)setSelectedIndexes:(NSIndexSet *)indexes { [collectionView setSelectionIndexes:indexes]; [collectionView scrollRectToVisible:[collectionView frameForItemAtIndex:indexes.firstIndex]]; } - (void)refresh { [items enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) { [[collectionView itemAtIndex:idx] setRepresentedObject:obj]; }]; } - (void)willLoadPath:(NSString *)path { isLoading = YES; [NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(showLoadingOverlay) object:nil]; [self performSelector:@selector(showLoadingOverlay) withObject:nil afterDelay:0.25f]; } - (void)showVersions:(P4Item *)item { NSInteger idx = [[collectionView selectionIndexes] firstIndex]; IconViewItem *iconItem = (IconViewItem *)[collectionView itemAtIndex:idx]; VersionsViewController *controller = [[VersionsViewController alloc] initWithItem:item actionDelegate:[self.view.window windowController]]; popover = [[NSPopover alloc] init]; [popover setBehavior:NSPopoverBehaviorTransient]; [popover setContentViewController:controller]; [popover showRelativeToRect:[iconItem.view frame] ofView:[iconItem.view superview] preferredEdge:NSMaxXEdge]; popover.delegate = self; [(PSCustomScrollView *)self.view setScrollDisabled:YES]; // [popover setBehavior:NSPopoverBehaviorApplicationDefined]; // NSWindow *popoverWindow = popover.contentViewController.view.window; // [NSApp runModalForWindow:popoverWindow]; } - (void)showFolderHistory:(P4Item *)item { NSInteger idx = [[collectionView selectionIndexes] firstIndex]; IconViewItem *iconItem = (IconViewItem *)[collectionView itemAtIndex:idx]; HistoryViewController *controller = [[HistoryViewController alloc] initWithDirectoryItem:item actionDelegate:[self.view.window windowController]]; popover = [[NSPopover alloc] init]; [popover setBehavior:NSPopoverBehaviorTransient]; [popover setContentViewController:controller]; [popover showRelativeToRect:[iconItem.view frame] ofView:[iconItem.view superview] preferredEdge:NSMaxXEdge]; popover.delegate = self; [(PSCustomScrollView *)self.view setScrollDisabled:YES]; // [popover setBehavior:NSPopoverBehaviorApplicationDefined]; // NSWindow *popoverWindow = popover.contentViewController.view.window; // [NSApp runModalForWindow:popoverWindow]; } - (void)editItemName:(P4Item*)directoryItem { NSInteger index = [items indexOfObjectIdenticalTo:directoryItem]; if (index == NSNotFound) return; [collectionView scrollRectToVisible:[collectionView frameForItemAtIndex:index]]; IconViewItem *viewItem = (IconViewItem *)[collectionView itemAtIndex:index]; P4Item *item = viewItem.representedObject; if (!item.isEditable) return; [viewItem beginEditing]; } - (void)setBackgroundColor:(NSColor *)color { #warning TODO } - (void)setDraggingSourceOperationMask:(NSDragOperation)mask forLocal:(BOOL)isLocal { [collectionView setDraggingSourceOperationMask:mask forLocal:isLocal]; } #pragma mark - NSPopoverDelegate - (void)popoverDidClose:(NSNotification *)notification { [(PSCustomScrollView *)self.view setScrollDisabled:NO]; } #pragma mark - IconViewItemDelegate - (void)iconViewItemSelected:(IconViewItem *)iconItem { } - (void)iconViewItemOpened:(IconViewItem *)iconItem { P4Item *item = iconItem.representedObject; if ([item isDirectory]) { [self setWorkingItem:item]; } else { // Perform default item action P4ItemAction *action = [item defaultAction]; action.delegate = [self.view.window windowController]; [action performAction]; } } - (void)iconViewItemDidEndEditing:(IconViewItem *)iconViewItem { [self renameItem:iconViewItem.representedObject name:iconViewItem.title]; } #pragma mark - P4Item delegate - (void)itemDidLoad:(P4Item *)item { [super itemDidLoad:item]; if (isLoading) { isLoading = NO; [NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(showLoadingOverlay) object:nil]; [loadingOverlay removeFromSuperview]; } else { if (item == workingItem) { // Did load children [self setItems:item.children]; } else if (item.parent == workingItem) { NSInteger idx = [items indexOfObjectIdenticalTo:item]; if (idx != NSNotFound) [[collectionView itemAtIndex:idx] setRepresentedObject:item]; } } } - (void)itemDidInvalidate:(P4Item *)item { } - (void)item:(id)item didFailWithError:(NSError *)error { [self failWithError:error]; [self itemDidLoad:item]; PSLog(@"IconView error: %@", error.localizedDescription); } #pragma mark - Quicklook Panel delegate - (BOOL)acceptsPreviewPanelControl:(QLPreviewPanel *)panel { return YES; } - (void)beginPreviewPanelControl:(QLPreviewPanel *)panel { quickLookPanel = panel; panel.delegate = self; panel.dataSource = self; } - (void)endPreviewPanelControl:(QLPreviewPanel *)panel { quickLookPanel = nil; } - (BOOL)previewPanel:(QLPreviewPanel *)panel handleEvent:(NSEvent *)event { // Redirect all key down events to the browser if ([event type] == NSKeyDown) { [collectionView keyDown:event]; return YES; } return NO; } - (NSInteger)numberOfPreviewItemsInPreviewPanel:(QLPreviewPanel *)panel { quickLookItems = [self selectedItems]; return quickLookItems.count; } - (id )previewPanel:(QLPreviewPanel *)panel previewItemAtIndex:(NSInteger)index { return [quickLookItems objectAtIndex:index]; } - (NSRect)previewPanel:(QLPreviewPanel *)panel sourceFrameOnScreenForPreviewItem:(id )item { NSIndexSet *indexes = [collectionView selectionIndexes]; if (!indexes.count) return CGRectZero; IconViewItem *iconItem = (id)[collectionView itemAtIndex:indexes.firstIndex]; quickLookIconImage = iconItem.iconView.image; // Get image frame CGRect imageRect = iconItem.iconView.frame; imageRect.origin = iconItem.iconView.superview.frame.origin; CGRect itemRect = [collectionView frameForItemAtIndex:indexes.firstIndex]; imageRect.origin.x += itemRect.origin.x; imageRect.origin.y += itemRect.origin.y; // Check that the icon rect is visible on screen CGRect visibleRect = [collectionView visibleRect]; if (!NSIntersectsRect(visibleRect, imageRect)) return CGRectZero; // Convert icon rect to screen coordinates imageRect = [collectionView convertRect:imageRect toView:nil]; imageRect = [collectionView.window convertRectToScreen:imageRect]; return imageRect; } - (id)previewPanel:(QLPreviewPanel *)panel transitionImageForPreviewItem:(id )item contentRect:(NSRect *)contentRect { return quickLookIconImage; } #pragma mark - Menu delegate - (void)menuNeedsUpdate:(NSMenu *)menu { [menu setAllowsContextMenuPlugIns:NO]; P4Item *item = nil; NSInteger index = [collectionView clickedIndex]; if (index == NSNotFound) { item = workingItem; [collectionView setSelectionIndexes:nil]; } else { NSIndexSet *set = [collectionView selectionIndexes]; if (set.count > 1 && [set containsIndex:index]) { // Multiple [menu setItems:[items objectsAtIndexes:set] delegate:[self.view.window windowController]]; return; } else { [collectionView setSelectionIndexes:[NSIndexSet indexSetWithIndex:index]]; item = [items objectAtIndex:index]; } } [menu setItem:item delegate:[self.view.window windowController]]; } @end