1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/platform_util.h" 6 7 #include <Carbon/Carbon.h> 8 #import <Cocoa/Cocoa.h> 9 #include <CoreServices/CoreServices.h> 10 11 #include "base/files/file_path.h" 12 #include "base/logging.h" 13 #include "base/mac/mac_logging.h" 14 #include "base/mac/scoped_aedesc.h" 15 #include "base/strings/sys_string_conversions.h" 16 #include "grit/generated_resources.h" 17 #include "ui/base/l10n/l10n_util.h" 18 #include "ui/base/l10n/l10n_util_mac.h" 19 #include "url/gurl.h" 20 21 namespace platform_util { 22 23 void ShowItemInFolder(const base::FilePath& full_path) { 24 DCHECK([NSThread isMainThread]); 25 NSString* path_string = base::SysUTF8ToNSString(full_path.value()); 26 if (!path_string || ![[NSWorkspace sharedWorkspace] selectFile:path_string 27 inFileViewerRootedAtPath:nil]) 28 LOG(WARNING) << "NSWorkspace failed to select file " << full_path.value(); 29 } 30 31 // This function opens a file. This doesn't use LaunchServices or NSWorkspace 32 // because of two bugs: 33 // 1. Incorrect app activation with com.apple.quarantine: 34 // http://crbug.com/32921 35 // 2. Silent no-op for unassociated file types: http://crbug.com/50263 36 // Instead, an AppleEvent is constructed to tell the Finder to open the 37 // document. 38 void OpenItem(const base::FilePath& full_path) { 39 DCHECK([NSThread isMainThread]); 40 NSString* path_string = base::SysUTF8ToNSString(full_path.value()); 41 if (!path_string) 42 return; 43 44 // Create the target of this AppleEvent, the Finder. 45 base::mac::ScopedAEDesc<AEAddressDesc> address; 46 const OSType finderCreatorCode = 'MACS'; 47 OSErr status = AECreateDesc(typeApplSignature, // type 48 &finderCreatorCode, // data 49 sizeof(finderCreatorCode), // dataSize 50 address.OutPointer()); // result 51 if (status != noErr) { 52 OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE target"; 53 return; 54 } 55 56 // Build the AppleEvent data structure that instructs Finder to open files. 57 base::mac::ScopedAEDesc<AppleEvent> theEvent; 58 status = AECreateAppleEvent(kCoreEventClass, // theAEEventClass 59 kAEOpenDocuments, // theAEEventID 60 address, // target 61 kAutoGenerateReturnID, // returnID 62 kAnyTransactionID, // transactionID 63 theEvent.OutPointer()); // result 64 if (status != noErr) { 65 OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE event"; 66 return; 67 } 68 69 // Create the list of files (only ever one) to open. 70 base::mac::ScopedAEDesc<AEDescList> fileList; 71 status = AECreateList(NULL, // factoringPtr 72 0, // factoredSize 73 false, // isRecord 74 fileList.OutPointer()); // resultList 75 if (status != noErr) { 76 OSSTATUS_LOG(WARNING, status) << "Could not create OpenItem() AE file list"; 77 return; 78 } 79 80 // Add the single path to the file list. C-style cast to avoid both a 81 // static_cast and a const_cast to get across the toll-free bridge. 82 CFURLRef pathURLRef = (CFURLRef)[NSURL fileURLWithPath:path_string]; 83 FSRef pathRef; 84 if (CFURLGetFSRef(pathURLRef, &pathRef)) { 85 status = AEPutPtr(fileList.OutPointer(), // theAEDescList 86 0, // index 87 typeFSRef, // typeCode 88 &pathRef, // dataPtr 89 sizeof(pathRef)); // dataSize 90 if (status != noErr) { 91 OSSTATUS_LOG(WARNING, status) 92 << "Could not add file path to AE list in OpenItem()"; 93 return; 94 } 95 } else { 96 LOG(WARNING) << "Could not get FSRef for path URL in OpenItem()"; 97 return; 98 } 99 100 // Attach the file list to the AppleEvent. 101 status = AEPutParamDesc(theEvent.OutPointer(), // theAppleEvent 102 keyDirectObject, // theAEKeyword 103 fileList); // theAEDesc 104 if (status != noErr) { 105 OSSTATUS_LOG(WARNING, status) 106 << "Could not put the AE file list the path in OpenItem()"; 107 return; 108 } 109 110 // Send the actual event. Do not care about the reply. 111 base::mac::ScopedAEDesc<AppleEvent> reply; 112 status = AESend(theEvent, // theAppleEvent 113 reply.OutPointer(), // reply 114 kAENoReply + kAEAlwaysInteract, // sendMode 115 kAENormalPriority, // sendPriority 116 kAEDefaultTimeout, // timeOutInTicks 117 NULL, // idleProc 118 NULL); // filterProc 119 if (status != noErr) { 120 OSSTATUS_LOG(WARNING, status) 121 << "Could not send AE to Finder in OpenItem()"; 122 } 123 } 124 125 void OpenExternal(const GURL& url) { 126 DCHECK([NSThread isMainThread]); 127 NSString* url_string = base::SysUTF8ToNSString(url.spec()); 128 NSURL* ns_url = [NSURL URLWithString:url_string]; 129 if (!ns_url || ![[NSWorkspace sharedWorkspace] openURL:ns_url]) 130 LOG(WARNING) << "NSWorkspace failed to open URL " << url; 131 } 132 133 gfx::NativeWindow GetTopLevel(gfx::NativeView view) { 134 return [view window]; 135 } 136 137 gfx::NativeView GetParent(gfx::NativeView view) { 138 return nil; 139 } 140 141 bool IsWindowActive(gfx::NativeWindow window) { 142 return [window isKeyWindow] || [window isMainWindow]; 143 } 144 145 void ActivateWindow(gfx::NativeWindow window) { 146 [window makeKeyAndOrderFront:nil]; 147 } 148 149 bool IsVisible(gfx::NativeView view) { 150 // A reasonable approximation of how you'd expect this to behave. 151 return (view && 152 ![view isHiddenOrHasHiddenAncestor] && 153 [view window] && 154 [[view window] isVisible]); 155 } 156 157 bool IsSwipeTrackingFromScrollEventsEnabled() { 158 SEL selector = @selector(isSwipeTrackingFromScrollEventsEnabled); 159 return [NSEvent respondsToSelector:selector] 160 && [NSEvent performSelector:selector]; 161 } 162 163 } // namespace platform_util 164