Index: chrome/browser/mac/install_from_dmg.mm |
=================================================================== |
--- chrome/browser/mac/install_from_dmg.mm (revision 91136) |
+++ chrome/browser/mac/install_from_dmg.mm (working copy) |
@@ -10,6 +10,7 @@ |
#include <CoreServices/CoreServices.h> |
#include <DiskArbitration/DiskArbitration.h> |
#include <IOKit/IOKitLib.h> |
+#include <signal.h> |
#include <stdlib.h> |
#include <string.h> |
#include <sys/param.h> |
@@ -29,6 +30,7 @@ |
#include "chrome/browser/mac/scoped_authorizationref.h" |
#include "chrome/browser/mac/scoped_ioobject.h" |
#import "chrome/browser/mac/keystone_glue.h" |
+#include "chrome/browser/mac/launchd.h" |
#include "chrome/browser/mac/relauncher.h" |
#include "chrome/common/chrome_constants.h" |
#include "grit/chromium_strings.h" |
@@ -47,6 +49,11 @@ |
namespace { |
+NSString* const kDockTileDataKey = @"tile-data"; |
+NSString* const kDockFileDataKey = @"file-data"; |
+NSString* const kDockCFURLStringKey = @"_CFURLString"; |
+NSString* const kDockCFURLStringTypeKey = @"_CFURLStringType"; |
+ |
// Given an io_service_t (expected to be of class IOMedia), walks the ancestor |
// chain, returning the closest ancestor that implements class IOHDIXHDDrive, |
// if any. If no such ancestor is found, returns NULL. Following the "copy" |
@@ -340,6 +347,294 @@ |
return true; |
} |
+// Returns an array parallel to |persistent_apps| containing only the |
+// pathnames of the Dock tiles contained therein. Returns nil on failure, such |
+// as when the structure of |persistent_apps| is not understood. |
+NSMutableArray* PersistentAppPaths(NSArray* persistent_apps) { |
+ NSMutableArray* app_paths = |
+ [NSMutableArray arrayWithCapacity:[persistent_apps count]]; |
+ |
+ for (NSDictionary* app in persistent_apps) { |
+ if (![app isKindOfClass:[NSDictionary class]]) { |
+ LOG(ERROR) << "app not NSDictionary"; |
+ return nil; |
+ } |
+ |
+ NSDictionary* tile_data = [app objectForKey:kDockTileDataKey]; |
+ if (![tile_data isKindOfClass:[NSDictionary class]]) { |
+ LOG(ERROR) << "tile_data not NSDictionary"; |
+ return nil; |
+ } |
+ |
+ NSDictionary* file_data = [tile_data objectForKey:kDockFileDataKey]; |
+ if (![file_data isKindOfClass:[NSDictionary class]]) { |
+ LOG(ERROR) << "file_data not NSDictionary"; |
+ return nil; |
+ } |
+ |
+ NSNumber* type = [file_data objectForKey:kDockCFURLStringTypeKey]; |
+ if (![type isKindOfClass:[NSNumber class]]) { |
+ LOG(ERROR) << "type not NSNumber"; |
+ return nil; |
+ } |
+ if ([type intValue] != 0) { |
+ LOG(ERROR) << "type not 0"; |
+ return nil; |
+ } |
+ |
+ NSString* path = [file_data objectForKey:kDockCFURLStringKey]; |
+ if (![path isKindOfClass:[NSString class]]) { |
+ LOG(ERROR) << "path not NSString"; |
+ return nil; |
+ } |
+ |
+ [app_paths addObject:path]; |
+ } |
+ |
+ return app_paths; |
+} |
+ |
+// Adds an icon to the Dock pointing to |installed_path| if one is not already |
+// present. |dmg_app_path| is the path to the install source. Its tile will be |
+// removed if present. If any changes are made to the Dock's configuration, |
+// the Dock process is restarted to reflect those changes. |
+// |
+// Various heruistics are used to determine where to place the new icon |
+// relative to other items already present in the Dock: |
+// - If installed_path is already in the Dock, no new tiles for this path |
+// will be added. |
+// - If dmg_app_path is present in the Dock, it will be removed. If |
+// installed_path is not already present, the new tile referencing |
+// installed_path will be placed where the dmg_app_path tile was. This |
+// keeps the tile where a user expects it if they dragged the application |
+// icon from a disk image into the Dock and then clicked on the new icon |
+// in the Dock. |
+// - The new tile will precede any application with the same name already |
+// in the Dock. |
+// - In an official build, a new tile for Google Chrome will be placed |
+// immediately before the first existing tile for Google Chrome Canary, |
+// and a new tile for Google Chrome Canary will be placed immediately after |
+// the last existing tile for Google Chrome. |
+// - The new tile will be placed immediately after the last tile for another |
+// browser application already in the Dock. |
+// - The new tile will be placed last in the Dock. |
+// For the purposes of these comparisons, applications are identified by the |
+// last component in their path. For example, any application named Safari.app |
+// will be treated as a browser. If the user renames an application on disk, |
+// it will alter the result. Looking up the bundle ID could be slightly more |
+// robust in the presence of such alterations, but it's not thought to be a |
+// large enough problem to warrant such lookups. |
+// |
+// The changes made to the Dock's configuration are the minimal changes |
+// necessary to cause the desired behavior. Although it's possible to set |
+// additional properties on the dock tile added to the Dock's plist, this |
+// is not done. Upon relaunch, Dock.app will determine the correct values for |
+// the properties it requires and add them to its configuration. |
+// |
+// ApplicationServices.framework/Frameworks/HIServices.framework contains an |
+// undocumented function, CoreDockAddFileToDock, that is able to add items to |
+// the Dock "live" without requiring a Dock restart. Under the hood, it |
+// communicates with the Dock via Mach IPC. It is available as of Mac OS X |
+// 10.6. AddDockIcon could call CoreDockAddFileToDock if available, but |
+// CoreDockAddFileToDock seems to always to add the new Dock icon last, where |
+// AddDockIcon takes care to position the icon appropriately. Based on |
+// disassembly, the signature of the undocumented function appears to be |
+// extern "C" OSStatus CoreDockAddFileToDock(CFURLRef url, int); |
+// The int argument doesn't appear to have any effect. It's not used as the |
+// position to place the icon as hoped. |
+void AddDockIcon(NSString* installed_path, NSString* dmg_app_path) { |
+ // There's enough potential allocation in this function to justify a |
+ // distinct pool. |
+ base::mac::ScopedNSAutoreleasePool autorelease_pool; |
+ |
+ NSString* const kDockDomain = @"com.apple.dock"; |
+ NSUserDefaults* user_defaults = [NSUserDefaults standardUserDefaults]; |
+ |
+ NSDictionary* dock_plist_const = |
+ [user_defaults persistentDomainForName:kDockDomain]; |
+ if (![dock_plist_const isKindOfClass:[NSDictionary class]]) { |
+ LOG(ERROR) << "dock_plist_const not NSDictionary"; |
+ return; |
+ } |
+ NSMutableDictionary* dock_plist = |
+ [NSMutableDictionary dictionaryWithDictionary:dock_plist_const]; |
+ |
+ NSString* const kDockPersistentAppsKey = @"persistent-apps"; |
+ NSArray* persistent_apps_const = |
+ [dock_plist objectForKey:kDockPersistentAppsKey]; |
+ if (![persistent_apps_const isKindOfClass:[NSArray class]]) { |
+ LOG(ERROR) << "persistent_apps_const not NSArray"; |
+ return; |
+ } |
+ NSMutableArray* persistent_apps = |
+ [NSMutableArray arrayWithArray:persistent_apps_const]; |
+ |
+ NSMutableArray* persistent_app_paths = PersistentAppPaths(persistent_apps); |
+ if (!persistent_app_paths) { |
+ return; |
+ } |
+ |
+ // Directories in the Dock's plist are given with trailing slashes. Since |
+ // installed_path and dmg_app_path both refer to application bundles, |
+ // they're directories and will show up with trailing slashes. This is an |
+ // artifact of the Dock's internal use of CFURL. Look for paths that match, |
+ // and when adding an item to the Dock's plist, keep it in the form that the |
+ // Dock likes. |
+ NSString* installed_path_dock = [installed_path stringByAppendingString:@"/"]; |
+ NSString* dmg_app_path_dock = [dmg_app_path stringByAppendingString:@"/"]; |
+ |
+ NSUInteger already_installed_app_index = NSNotFound; |
+ NSUInteger app_index = NSNotFound; |
+ for (NSUInteger index = 0; index < [persistent_apps count]; ++index) { |
+ NSString* app_path = [persistent_app_paths objectAtIndex:index]; |
+ if ([app_path isEqualToString:installed_path_dock]) { |
+ // If the Dock already contains a reference to the newly installed |
+ // application, don't add another one. |
+ already_installed_app_index = index; |
+ } else if ([app_path isEqualToString:dmg_app_path_dock]) { |
+ // If the Dock contains a reference to the application on the disk |
+ // image, replace it with a reference to the newly installed |
+ // application. However, if the Dock contains a reference to both the |
+ // application on the disk image and the newly installed application, |
+ // just remove the one referencing the disk image. |
+ // |
+ // This case is only encountered when the user drags the icon from the |
+ // disk image volume window in the Finder directly into the Dock. |
+ app_index = index; |
+ } |
+ } |
+ |
+ bool made_change = false; |
+ |
+ if (app_index != NSNotFound) { |
+ // Remove the Dock's reference to the application on the disk image. |
+ [persistent_apps removeObjectAtIndex:app_index]; |
+ [persistent_app_paths removeObjectAtIndex:app_index]; |
+ made_change = true; |
+ } |
+ |
+ if (already_installed_app_index == NSNotFound) { |
+ // The Dock doesn't yet have a reference to the icon at the |
+ // newly installed path. Figure out where to put the new icon. |
+ NSString* app_name = [installed_path lastPathComponent]; |
+ |
+ if (app_index == NSNotFound) { |
+ // If an application with this name is already in the Dock, put the new |
+ // one right before it. |
+ for (NSUInteger index = 0; index < [persistent_apps count]; ++index) { |
+ NSString* dock_app_name = |
+ [[persistent_app_paths objectAtIndex:index] lastPathComponent]; |
+ if ([dock_app_name isEqualToString:app_name]) { |
+ app_index = index; |
+ break; |
+ } |
+ } |
+ } |
+ |
+#if defined(GOOGLE_CHROME_BUILD) |
+ if (app_index == NSNotFound) { |
+ // If this is an officially-branded Chrome (including Canary) and an |
+ // application matching the "other" flavor is already in the Dock, put |
+ // them next to each other. Google Chrome will precede Google Chrome |
+ // Canary in the Dock. |
+ NSString* chrome_name = @"Google Chrome.app"; |
+ NSString* canary_name = @"Google Chrome Canary.app"; |
+ for (NSUInteger index = 0; index < [persistent_apps count]; ++index) { |
+ NSString* dock_app_name = |
+ [[persistent_app_paths objectAtIndex:index] lastPathComponent]; |
+ if ([dock_app_name isEqualToString:canary_name] && |
+ [app_name isEqualToString:chrome_name]) { |
+ app_index = index; |
+ |
+ // Break: put Google Chrome.app before the first Google Chrome |
+ // Canary.app. |
+ break; |
+ } else if ([dock_app_name isEqualToString:chrome_name] && |
+ [app_name isEqualToString:canary_name]) { |
+ app_index = index + 1; |
+ |
+ // No break: put Google Chrome Canary.app after the last Google |
+ // Chrome.app. |
+ } |
+ } |
+ } |
+#endif // GOOGLE_CHROME_BUILD |
+ |
+ if (app_index == NSNotFound) { |
+ // Put the new application after the last browser application already |
+ // present in the Dock. |
+ NSArray* other_browser_app_names = |
+ [NSArray arrayWithObjects: |
+#if defined(GOOGLE_CHROME_BUILD) |
+ @"Chromium.app", // Unbranded Google Chrome |
+#else |
+ @"Google Chrome.app", |
+ @"Google Chrome Canary.app", |
+#endif |
+ @"Safari.app", |
+ @"Firefox.app", |
+ @"Camino.app", |
+ @"Opera.app", |
+ @"OmniWeb.app", |
+ @"WebKit.app", // Safari nightly |
+ @"Aurora.app", // Firefox dev |
+ @"Nightly.app", // Firefox nightly |
+ nil]; |
+ for (NSUInteger index = 0; index < [persistent_apps count]; ++index) { |
+ NSString* dock_app_name = |
+ [[persistent_app_paths objectAtIndex:index] lastPathComponent]; |
+ if ([other_browser_app_names containsObject:dock_app_name]) { |
+ app_index = index + 1; |
+ } |
+ } |
+ } |
+ |
+ if (app_index == NSNotFound) { |
+ // Put the new application last in the Dock. |
+ app_index = [persistent_apps count]; |
+ } |
+ |
+ // Set up the new Dock tile. |
+ NSDictionary* new_tile_file_data = |
+ [NSDictionary dictionaryWithObjectsAndKeys: |
+ installed_path_dock, kDockCFURLStringKey, |
+ [NSNumber numberWithInt:0], kDockCFURLStringTypeKey, |
+ nil]; |
+ NSDictionary* new_tile_data = |
+ [NSDictionary dictionaryWithObject:new_tile_file_data |
+ forKey:kDockFileDataKey]; |
+ NSDictionary* new_tile = |
+ [NSDictionary dictionaryWithObject:new_tile_data |
+ forKey:kDockTileDataKey]; |
+ |
+ // Add the new tile to the Dock. |
+ [persistent_apps insertObject:new_tile atIndex:app_index]; |
+ [persistent_app_paths insertObject:installed_path_dock atIndex:app_index]; |
+ made_change = true; |
+ } |
+ |
+ // Verify that the arrays are still parallel. |
+ DCHECK_EQ([persistent_apps count], [persistent_app_paths count]); |
+ |
+ if (!made_change) { |
+ // If no changes were made, there's no point in rewriting the Dock's |
+ // plist or restarting the Dock. |
+ return; |
+ } |
+ |
+ // Rewrite the plist. |
+ [dock_plist setObject:persistent_apps forKey:kDockPersistentAppsKey]; |
+ [user_defaults setPersistentDomain:dock_plist forName:kDockDomain]; |
+ |
+ // Restart the Dock. Doing this via launchd using the proper job label is |
+ // the safest way to handle the restart. Unlike "killall Dock", looking this |
+ // up via launchd guarantees that only the right process will be targeted. |
+ // Sending a SIGHUP to the Dock seems to be a more reliable way to get the |
+ // replacement Dock process to read the newly written plist than using the |
+ // equivalent of "launchctl stop" (even if followed by "launchctl start.") |
+ launchd::SignalJob("com.apple.Dock.agent", SIGHUP); |
+} |
+ |
// Launches the application at installed_path. The helper application |
// contained within install_path will be used for the relauncher process. This |
// keeps Launch Services from ever having to see or think about the helper |
@@ -448,12 +743,18 @@ |
if (!InstallFromDiskImage(authorization.release(), |
installer_path, |
source_path, |
- target_path) || |
- !LaunchInstalledApp(target_path, dmg_bsd_device_name)) { |
+ target_path)) { |
ShowErrorDialog(); |
return false; |
} |
+ AddDockIcon(target_path, source_path); |
+ |
+ if (!LaunchInstalledApp(target_path, dmg_bsd_device_name)) { |
+ ShowErrorDialog(); |
+ return false; |
+ } |
+ |
return true; |
} |