// // AppDelegate.m // Perforce // // Created by Adam Czubernat on 07.05.2013. // Copyright (c) 2013 Perforce Software, Inc. All rights reserved. // #import "AppDelegate.h" #import "P4Workspace.h" #import "P4Defaults.h" #import "P4WorkspaceDefaults.h" #import "MainWindowController.h" @interface AppDelegate () { MainWindowController *windowController; SEL launchAction; id launchObject; BOOL loggingIn; } - (void)automaticLogin; - (void)showLoginPanel; - (void)loadWorkspace; - (void)performWorkspaceAction:(SEL)selector withObject:(id)object; - (void)handleOpenEvent:(NSAppleEventDescriptor *)event withReplyEvent:(NSAppleEventDescriptor *)replyEvent; - (void)handleServices:(NSPasteboard *)pboard userData:(NSString *)userData error:(NSString **)error; @end @implementation AppDelegate - (void)applicationWillFinishLaunching:(NSNotification *)notification { // Set URL handler NSAppleEventManager *appleEventManager = [NSAppleEventManager sharedAppleEventManager]; [appleEventManager setEventHandler:self andSelector:@selector(handleOpenEvent:withReplyEvent:) forEventClass:kInternetEventClass andEventID:kAEGetURL]; // Register as default URL scheme handler CFStringRef bundleID = (__bridge CFStringRef)[[NSBundle mainBundle] bundleIdentifier]; LSSetDefaultHandlerForURLScheme((__bridge CFStringRef)@"p4", bundleID); } - (void)applicationDidFinishLaunching:(NSNotification *)notification { // Check if launched as service NSAppleEventDescriptor *event = [[NSAppleEventManager sharedAppleEventManager] currentAppleEvent]; BOOL launchedAsService = [event eventID] == kAEOpenApplication && [[event paramDescriptorForKeyword:keyAEPropData] enumCodeValue] == keyAELaunchedAsServiceItem; // Update services [NSApp setServicesProvider:self]; NSUpdateDynamicServices(); // Set the exception handler PSDebugInstallCrashHandler(); // Log app info NSDictionary *info = [[NSBundle mainBundle] infoDictionary]; NSString *appName = [info objectForKey:@"CFBundleExecutable"]; NSString *appVersion = [info objectForKey:@"CFBundleShortVersionString"]; NSString *appBuild = [info objectForKey:@"CFBundleVersion"]; NSString *appBuildDate = [info objectForKey:@"BuildDate"]; PSLogStore(@"AppVersion", @"%@ v%@ (%@)", appName, appVersion, appBuild); PSLogStore(@"BuildDate", @"%@", appBuildDate); // Load default values [NSUserDefaults loadDefaultsWithName:@"UserDefaults"]; NSString *lastWorkspace = [P4Defaults sharedInstance].workspace; windowController = [[MainWindowController alloc] init]; // Check if show window if (launchedAsService) { PSLog(@"Launching with service"); [windowController window]; } else { PSLog(@"Launching normally"); [windowController showWindow:nil]; } if (lastWorkspace.length) { [self automaticLogin]; if ([windowController.window isVisible]) [windowController showLoading:@"Connecting..."]; } else { [windowController showWindow:nil]; [self showLoginPanel]; } } //- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)sender { // return YES; //} - (BOOL)applicationShouldHandleReopen:(NSApplication *)sender hasVisibleWindows:(BOOL)windows { if (!windows) { [windowController showWindow:self]; if (loggingIn) [windowController showLoading:@"Connecting..."]; } return YES; } #pragma mark - Private - (void)automaticLogin { PSLog(@"Automatic login..."); loggingIn = YES; P4Credentials *credentials = [[P4Credentials alloc] init]; credentials.username = [P4Defaults sharedInstance].username; credentials.address = [P4Defaults sharedInstance].host; credentials.resumeSession = YES; // Connect to host [[P4Workspace sharedInstance] connectWithCredentials:credentials response:^(P4Operation *operation, NSArray *response) { if (operation.errors) { PSLog(@"Failed to connect"); [self showLoginPanel]; return; } // Try login last session [[P4Workspace sharedInstance] loginWithCredentials:credentials response:^(P4Operation *operation, NSArray *response) { if (operation.errors) { PSLog(@"Failed logging-in using last session"); PSLog(@"Automatic login using SSO..."); // Try SSO login [[P4Workspace sharedInstance] loginWithSSO:credentials response:^(P4Operation *operation, NSArray *response) { if (operation.errors) { PSLog(@"Failed to login using SSO"); [self showLoginPanel]; return; } PSLog(@"Logged-in using SSO"); [self loadWorkspace]; }]; return; } PSLog(@"Logged-in using last session"); [self loadWorkspace]; }]; }]; } - (void)showLoginPanel { loggingIn = NO; [windowController dismissLoading]; if (![windowController.window isVisible]) [windowController showWindow:nil]; [windowController connect:nil]; } - (void)loadWorkspace { NSString *workspace = [P4Defaults sharedInstance].workspace; [[P4Workspace sharedInstance] setWorkspace:workspace response:^(P4Operation *operation, NSArray *response) { loggingIn = NO; [windowController dismissLoading]; if (operation.errors) { PSLog(@"Couldn't use last workspace %@", workspace); [self showLoginPanel]; [windowController showWorkspaceSelection:nil]; } else { PSLog(@"Connected to last workspace: %@", workspace); PSLogStore(@"Workspace info", @"%@", response); // Save last workspace details [P4WorkspaceDefaults setWorkspace:workspace]; [windowController reload]; if (launchAction != @selector(openFile:)) [windowController loadPath:@"//"]; // Perform dispatched service action if (launchAction) { PSLog(@"Service: dispatched action -%@ %@", NSStringFromSelector(launchAction), launchObject); [windowController performSelectorOnMainThread:launchAction withObject:launchObject waitUntilDone:NO]; launchAction = nil; launchObject = nil; } } }]; } - (void)performWorkspaceAction:(SEL)selector withObject:(id)object { if ([[P4Workspace sharedInstance] isLoggedIn] && [[P4Workspace sharedInstance] workspace]) { PSLog(@"Service: action -%@ %@", NSStringFromSelector(selector), object); [windowController performSelectorOnMainThread:selector withObject:object waitUntilDone:NO]; } else { launchAction = selector; launchObject = object; } } #pragma mark - URL Event handler - (void)handleOpenEvent:(NSAppleEventDescriptor *)event withReplyEvent:(NSAppleEventDescriptor *)replyEvent { NSString *url = [[event paramDescriptorForKeyword:keyDirectObject] stringValue]; PSLog(@"Opening external URL:\n%@", url); [self performWorkspaceAction:@selector(openFile:) withObject:url]; } - (void)handleServices:(NSPasteboard *)pboard userData:(NSString *)userData error:(NSString *__autoreleasing *)error { PSLog(@"Service: %@", userData); NSArray *files = nil; if ([[pboard types] containsObject:NSFilenamesPboardType]){ files = [pboard propertyListForType:NSFilenamesPboardType]; } else if ([userData hasPrefix:@"com.adobe.illustrator"]) { NSAppleScript *appleScript = [[NSAppleScript alloc] initWithSource: @"tell application id \"com.adobe.illustrator\"\n" "set theFilePath to file path of front document\n" "end tell"]; NSDictionary *errorDict = nil; NSAppleEventDescriptor *descriptor = [appleScript executeAndReturnError:&errorDict]; NSURL *url = [NSURL URLWithString:[descriptor stringValue]]; NSString *path = [url path]; if(path) { files = [NSArray arrayWithObject:path]; } else { files = [NSArray array]; } if (!files.count) { PSLog(@"Error > Failed to run illustrator script %@\n%@", userData, errorDict); [windowController showErrors:@[ [NSError errorWithFormat:@"Failed to run illustrator script %@\n%@.\nIllustrator failed to return \nCheck if your file name contains illegal characters.", userData, errorDict] ]]; } } if (!files.count) return PSLog(@"Error > Failed to retrieve urls from service %@ %@", userData, pboard); // Parse urls if ([userData hasPrefix:@"com.apple.finder.folder"]) { // Add '/' to directory paths files = [files valueForKey:@"directoryPath"]; } if ([userData hasPrefix:@"com.apple.finder.symlink"]) { // Resolve symlinks files = [files arrayUsingBlock:^id(NSString *file, NSUInteger idx) { NSURL *url = [NSURL fileURLWithPath:file]; NSNumber *isSymlink = nil; [url getResourceValue:&isSymlink forKey:NSURLIsSymbolicLinkKey error:NULL]; if (isSymlink.boolValue) { NSString *dest = [[NSFileManager defaultManager] destinationOfSymbolicLinkAtPath:url.path error:NULL]; url = [NSURL fileURLWithPath:dest]; } NSNumber *isDir = nil; [url getResourceValue:&isDir forKey:NSURLIsDirectoryKey error:NULL]; if (isDir.boolValue) return [url.path directoryPath]; return url.path; }]; } if ([userData hasSuffix:@"edit"]) [self performWorkspaceAction:[files.firstObject hasSuffix:@"/"] ? @selector(editAllFiles:) : @selector(editFiles:) withObject:files]; else if ([userData hasSuffix:@"submit"]) [self performWorkspaceAction:[files.firstObject hasSuffix:@"/"] ? @selector(submitAllFiles:) : @selector(submitFiles:) withObject:files]; else if ([userData hasSuffix:@"open"]) [self performWorkspaceAction:@selector(openFile:) withObject:files.firstObject]; else PSLog(@"Error > Unrecognized service %@", userData); } @end
# | Change | User | Description | Committed | |
---|---|---|---|---|---|
#2 | 18548 | Robert Cowham | Merge from Main. | ||
#1 | 16507 | perforce_software | Move to main branch. | ||
//guest/perforce_software/piper/mac/R2.0/Perforce/AppDelegate.m | |||||
#1 | 12962 | alan_petersen |
Populate -o //guest/perforce_software/piper/mac/main/... //guest/perforce_software/piper/mac/R2.0/.... |
||
//guest/perforce_software/piper/mac/main/Perforce/AppDelegate.m | |||||
#2 | 12961 | alan_petersen |
Piper 2.0 Mega Update New Features/Functionality - Added help menu redirecting to URL. - Added readonly property for creating new workspaces. - Added html hyperlinks for Copy link functionality. - Added functionality for managing Finder Favorite items in sidebar. - Redesigned the way mapping is stored in Piper. - First version of syncing finder sidebar items with workspace mapping. - Small sorting improvements. - Creating Projects directory inside users home folder. - Adding Projects folder to finder sidebar item. - Creating and removing symbolic links accordingly to mapped folders. - Preventing duplicate names in symbolic links. - Refreshing symbolic links on mapping change inside application. - Storing workspace and server details in p4 configuration for other applications to use. - Added contextual menu items for Finder integration. - Added services menu for Adobe Illustrator integration. - Keyboard shortcuts for Illustrator integration. - Code refactoring and fixes for mapping issues. - Added Finder functionality to edit all files in folder. - Added user friendly message when editing a file using Finder outside the workspace. - Implemented hidden automatic login when opening application using Finder integration. - Logging to file in ~/Library/Logs - Unified workspace and all files views to show both local and depot files and folders. - Removed my workspace view references and logic. - Editing unmapped files on server. - First version of adding file to unmapped folders. - Showing opened by and edit actions in column details for all depot files. - Improved mappings functionality. - Enabled same feature options for mapped and unmapped folders and files. - Redesigned from scratch mapping and unmapping procedures for adding and removing files. - Implemented cleaning workspace using new mapping functionality. Removed debug overlay coloring. - Automated workspace creation - Improvements in editing files already mapped to workspace. - Implemented deleting remote files. - Implemented first version of move operation for remote files. - Removing last workspace information when disconnecting from workspace using app menu. - Implemented editing and submitting using symbolic links in project folder. New finder menu service for symbolic links Show in Piper which acts like share link functionality. - New icons for files and folders not tracked in the filesystem. - Improvements in showing file using share link. - Switched to new way of retrieving files in order to show user changes. - Redesigned and implemented new functionality for chaining operations with mapping. - Improvements and redesign of Edit/add actions to use new chaining logic . Fixed issue with file edit. - Improvements in window showing when using services. - Simplified file loading so the local files appears only when remote are also loaded. - Improved deleting of untracked files to avoid mapping and marking for delete. - Enabling simple copy paste and moving of remote and local files. - Added abort for exception handling in order to force crashing application on critical failures - Added custom exception handling for catching runtime errors to log and crash instead of continuing in unstable state. - Changed file copying to use mark for add . - Simplified and fixed responding file representations to mapping changes. Bug Fixes - Fixed crash when synchronizing. - Fixed sync issue when downloading directory without file size information. - Fixed issue with unread list crashing when file is not existing on disk. - Fixed incorrect sync progress calculation. - Removed relative path issues. - Fixed many of case-sensitivity problems. - Fixed deprecated methods and related issues in OS X 10.10. - Fixed folder rename not updating in column view. Revised and fixed many potential problems from implicit casting. - Fixed missing sync button on fast sync completion. - Refreshing mapping on synchronization. Fixed symbolic links not appearing until app is restarted. - Fixed latest crashing of autosync. - Fixed loading indicator issues. - Fixed and redesigned submit dialog to work correctly with Submit All Files option in Finder. - Fixed multiple error messages on network outage. Redesigned showing errors in main window. - Fixed opening random locations when using Finder integration. - Fixed issue when panel was detached from parent window. - Fixed bug when creating new workspace wouldn't store default settings. - Fixed memory issues with network operations. - Fixes in relogging mappings and file listing. - Improvements in editing unmapped files. - Fixed crash when adding file outside workspace. - Fixed breadcrumbs control issue. - Fixed issue with double parent folders when opening unmapped files. - Fixed crashes on sync after mapping new files. - Fixed issue with editing file using Finder -- Merging code and additional fixes in add button functionality. - Fixed unsync not working - Fixed submit panel issue not selecting files with different name case. - Fixed missing revert and sync to workspace actions in some cases. - Fixed issue with Submit and Edit finder actions. Improvements in stability of finder integration. - Fixed issue with unsubmitted folders breaking status of files inside. - Fixed issue with added files not showing correct icon and status. - Fixed bug with file edit resulting in a new directory named exactly like a file. - Fixed issue with reloading of subpath resulting in untracked folders. - Fixed mapping issue when result was always view mapping not relative. - Fixed submit panel showing more than once. - Fixed illustrator services not working. - Fixed userdefaults preferences problem with workspace name being null. - Fixed userdefaults keypath problem of dot-containing workspace names. - Forcing recreating of browser to possibly prevent pre-10.10 errors with automatic workspace selection. - Fixed adding file to depot not presenting correct icon. - Fixed issues with reverting a file that was marked for add. - Presenting error when trying to submit untracked files. - Fixed issue when submit files service crashed when using unmapped files. - Fixed file representation disappearing when removing file. - Fixed issue with symlinks resolving working on 10.10 only. Issue related to workspace selection not showing. - Fixed error panel method calls unavailable in Mac OS versions before 10.10. Issue related to hanging error panels. - Fixed removing a local file resulting in action progress freezing. - Fixed open file not working after edit. - Fixing crash when mapping changed. Issue related to moving local file to unmapped folder and other similar cases. |
||
#1 | 11252 | alan_petersen | Rename/move file(s) | ||
//guest/perforce_software/piper/mac/Perforce/AppDelegate.m | |||||
#1 | 10744 | alan_petersen | Rename/move file(s) | ||
//guest/perforce_software/piper/Perforce/AppDelegate.m | |||||
#1 | 8919 | Matt Attaway | Initial add of Piper, a lightweight Perforce client for artists and designers. |