Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2604)

Unified Diff: base/mac_util.mm

Issue 6046009: Move base/mac_util.h to base/mac and use the base::mac namespace.... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: '' Created 9 years, 12 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « base/mac_util.h ('k') | base/mac_util_unittest.mm » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: base/mac_util.mm
===================================================================
--- base/mac_util.mm (revision 70358)
+++ base/mac_util.mm (working copy)
@@ -1,716 +0,0 @@
-// Copyright (c) 2010 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "base/mac_util.h"
-
-#import <Cocoa/Cocoa.h>
-
-#include "base/file_path.h"
-#include "base/logging.h"
-#include "base/mac/scoped_cftyperef.h"
-#include "base/message_loop.h"
-#include "base/scoped_nsobject.h"
-#include "base/sys_string_conversions.h"
-
-using base::mac::ScopedCFTypeRef;
-
-namespace {
-
-// a count of currently outstanding requests for full screen mode from browser
-// windows, plugins, etc.
-int g_full_screen_requests[mac_util::kNumFullScreenModes] = { 0, 0, 0};
-
-// Sets the appropriate SystemUIMode based on the current full screen requests.
-// Since only one SystemUIMode can be active at a given time, full screen
-// requests are ordered by priority. If there are no outstanding full screen
-// requests, reverts to normal mode. If the correct SystemUIMode is already
-// set, does nothing.
-void SetUIMode() {
- // Get the current UI mode.
- SystemUIMode current_mode;
- GetSystemUIMode(&current_mode, NULL);
-
- // Determine which mode should be active, based on which requests are
- // currently outstanding. More permissive requests take precedence. For
- // example, plugins request |kFullScreenModeAutoHideAll|, while browser
- // windows request |kFullScreenModeHideDock| when the fullscreen overlay is
- // down. Precedence goes to plugins in this case, so AutoHideAll wins over
- // HideDock.
- SystemUIMode desired_mode = kUIModeNormal;
- SystemUIOptions desired_options = 0;
- if (g_full_screen_requests[mac_util::kFullScreenModeAutoHideAll] > 0) {
- desired_mode = kUIModeAllHidden;
- desired_options = kUIOptionAutoShowMenuBar;
- } else if (g_full_screen_requests[mac_util::kFullScreenModeHideDock] > 0) {
- desired_mode = kUIModeContentHidden;
- } else if (g_full_screen_requests[mac_util::kFullScreenModeHideAll] > 0) {
- desired_mode = kUIModeAllHidden;
- }
-
- if (current_mode != desired_mode)
- SetSystemUIMode(desired_mode, desired_options);
-}
-
-bool WasLaunchedAsLoginItem() {
- ProcessSerialNumber psn = { 0, kCurrentProcess };
-
- scoped_nsobject<NSDictionary> process_info(
- mac_util::CFToNSCast(ProcessInformationCopyDictionary(&psn,
- kProcessDictionaryIncludeAllInformationMask)));
-
- long long temp = [[process_info objectForKey:@"ParentPSN"] longLongValue];
- ProcessSerialNumber parent_psn =
- { (temp >> 32) & 0x00000000FFFFFFFFLL, temp & 0x00000000FFFFFFFFLL };
-
- scoped_nsobject<NSDictionary> parent_info(
- mac_util::CFToNSCast(ProcessInformationCopyDictionary(&parent_psn,
- kProcessDictionaryIncludeAllInformationMask)));
-
- // Check that creator process code is that of loginwindow.
- BOOL result =
- [[parent_info objectForKey:@"FileCreator"] isEqualToString:@"lgnw"];
-
- return result == YES;
-}
-
-// Looks into Shared File Lists corresponding to Login Items for the item
-// representing the current application. If such an item is found, returns
-// retained reference to it. Caller is responsible for releasing the reference.
-LSSharedFileListItemRef GetLoginItemForApp() {
- ScopedCFTypeRef<LSSharedFileListRef> login_items(LSSharedFileListCreate(
- NULL, kLSSharedFileListSessionLoginItems, NULL));
-
- if (!login_items.get()) {
- LOG(ERROR) << "Couldn't get a Login Items list.";
- return NULL;
- }
-
- scoped_nsobject<NSArray> login_items_array(
- mac_util::CFToNSCast(LSSharedFileListCopySnapshot(login_items, NULL)));
-
- NSURL* url = [NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]];
-
- for(NSUInteger i = 0; i < [login_items_array count]; ++i) {
- LSSharedFileListItemRef item = reinterpret_cast<LSSharedFileListItemRef>(
- [login_items_array objectAtIndex:i]);
- CFURLRef item_url_ref = NULL;
-
- if (LSSharedFileListItemResolve(item, 0, &item_url_ref, NULL) == noErr) {
- ScopedCFTypeRef<CFURLRef> item_url(item_url_ref);
- if (CFEqual(item_url, url)) {
- CFRetain(item);
- return item;
- }
- }
- }
-
- return NULL;
-}
-
-#if !defined(MAC_OS_X_VERSION_10_6) || \
- MAC_OS_X_VERSION_MAX_ALLOWED < MAC_OS_X_VERSION_10_6
-// kLSSharedFileListLoginItemHidden is supported on
-// 10.5, but missing from the 10.5 headers.
-// http://openradar.appspot.com/6482251
-static NSString* kLSSharedFileListLoginItemHidden =
- @"com.apple.loginitem.HideOnLaunch";
-#endif
-
-bool IsHiddenLoginItem(LSSharedFileListItemRef item) {
- ScopedCFTypeRef<CFBooleanRef> hidden(reinterpret_cast<CFBooleanRef>(
- LSSharedFileListItemCopyProperty(item,
- reinterpret_cast<CFStringRef>(kLSSharedFileListLoginItemHidden))));
-
- return hidden && hidden == kCFBooleanTrue;
-}
-
-} // end namespace
-
-namespace mac_util {
-
-std::string PathFromFSRef(const FSRef& ref) {
- ScopedCFTypeRef<CFURLRef> url(
- CFURLCreateFromFSRef(kCFAllocatorDefault, &ref));
- NSString *path_string = [(NSURL *)url.get() path];
- return [path_string fileSystemRepresentation];
-}
-
-bool FSRefFromPath(const std::string& path, FSRef* ref) {
- OSStatus status = FSPathMakeRef((const UInt8*)path.c_str(),
- ref, nil);
- return status == noErr;
-}
-
-static bool g_override_am_i_bundled = false;
-static bool g_override_am_i_bundled_value = false;
-
-// Adapted from http://developer.apple.com/carbon/tipsandtricks.html#AmIBundled
-static bool UncachedAmIBundled() {
- if (g_override_am_i_bundled)
- return g_override_am_i_bundled_value;
-
- ProcessSerialNumber psn = {0, kCurrentProcess};
-
- FSRef fsref;
- OSStatus pbErr;
- if ((pbErr = GetProcessBundleLocation(&psn, &fsref)) != noErr) {
- LOG(ERROR) << "GetProcessBundleLocation failed: error " << pbErr;
- return false;
- }
-
- FSCatalogInfo info;
- OSErr fsErr;
- if ((fsErr = FSGetCatalogInfo(&fsref, kFSCatInfoNodeFlags, &info,
- NULL, NULL, NULL)) != noErr) {
- LOG(ERROR) << "FSGetCatalogInfo failed: error " << fsErr;
- return false;
- }
-
- return info.nodeFlags & kFSNodeIsDirectoryMask;
-}
-
-bool AmIBundled() {
- // If the return value is not cached, this function will return different
- // values depending on when it's called. This confuses some client code, see
- // http://crbug.com/63183 .
- static bool result = UncachedAmIBundled();
- DCHECK_EQ(result, UncachedAmIBundled())
- << "The return value of AmIBundled() changed. This will confuse tests. "
- << "Call SetAmIBundled() override manually if your test binary "
- << "delay-loads the framework.";
- return result;
-}
-
-void SetOverrideAmIBundled(bool value) {
- g_override_am_i_bundled = true;
- g_override_am_i_bundled_value = value;
-}
-
-bool IsBackgroundOnlyProcess() {
- // This function really does want to examine NSBundle's idea of the main
- // bundle dictionary, and not the overriden MainAppBundle. It needs to look
- // at the actual running .app's Info.plist to access its LSUIElement
- // property.
- NSDictionary* info_dictionary = [[NSBundle mainBundle] infoDictionary];
- return [[info_dictionary objectForKey:@"LSUIElement"] boolValue] != NO;
-}
-
-// No threading worries since NSBundle isn't thread safe.
-static NSBundle* g_override_app_bundle = nil;
-
-NSBundle* MainAppBundle() {
- if (g_override_app_bundle)
- return g_override_app_bundle;
- return [NSBundle mainBundle];
-}
-
-FilePath MainAppBundlePath() {
- NSBundle* bundle = MainAppBundle();
- return FilePath([[bundle bundlePath] fileSystemRepresentation]);
-}
-
-void SetOverrideAppBundle(NSBundle* bundle) {
- if (bundle != g_override_app_bundle) {
- [g_override_app_bundle release];
- g_override_app_bundle = [bundle retain];
- }
-}
-
-void SetOverrideAppBundlePath(const FilePath& file_path) {
- NSString* path = base::SysUTF8ToNSString(file_path.value());
- NSBundle* bundle = [NSBundle bundleWithPath:path];
- CHECK(bundle) << "Failed to load the bundle at " << file_path.value();
-
- SetOverrideAppBundle(bundle);
-}
-
-OSType CreatorCodeForCFBundleRef(CFBundleRef bundle) {
- OSType creator = kUnknownType;
- CFBundleGetPackageInfo(bundle, NULL, &creator);
- return creator;
-}
-
-OSType CreatorCodeForApplication() {
- CFBundleRef bundle = CFBundleGetMainBundle();
- if (!bundle)
- return kUnknownType;
-
- return CreatorCodeForCFBundleRef(bundle);
-}
-
-bool GetSearchPathDirectory(NSSearchPathDirectory directory,
- NSSearchPathDomainMask domain_mask,
- FilePath* result) {
- DCHECK(result);
- NSArray* dirs =
- NSSearchPathForDirectoriesInDomains(directory, domain_mask, YES);
- if ([dirs count] < 1) {
- return false;
- }
- NSString* path = [dirs objectAtIndex:0];
- *result = FilePath([path fileSystemRepresentation]);
- return true;
-}
-
-bool GetLocalDirectory(NSSearchPathDirectory directory, FilePath* result) {
- return GetSearchPathDirectory(directory, NSLocalDomainMask, result);
-}
-
-bool GetUserDirectory(NSSearchPathDirectory directory, FilePath* result) {
- return GetSearchPathDirectory(directory, NSUserDomainMask, result);
-}
-
-FilePath GetUserLibraryPath() {
- FilePath user_library_path;
- if (!GetUserDirectory(NSLibraryDirectory, &user_library_path)) {
- LOG(WARNING) << "Could not get user library path";
- }
- return user_library_path;
-}
-
-CGColorSpaceRef GetSRGBColorSpace() {
- // Leaked. That's OK, it's scoped to the lifetime of the application.
- static CGColorSpaceRef g_color_space_sRGB =
- CGColorSpaceCreateWithName(kCGColorSpaceSRGB);
- LOG_IF(ERROR, !g_color_space_sRGB) << "Couldn't get the sRGB color space";
- return g_color_space_sRGB;
-}
-
-CGColorSpaceRef GetSystemColorSpace() {
- // Leaked. That's OK, it's scoped to the lifetime of the application.
- // Try to get the main display's color space.
- static CGColorSpaceRef g_system_color_space =
- CGDisplayCopyColorSpace(CGMainDisplayID());
-
- if (!g_system_color_space) {
- // Use a generic RGB color space. This is better than nothing.
- g_system_color_space = CGColorSpaceCreateDeviceRGB();
-
- if (g_system_color_space) {
- LOG(WARNING) <<
- "Couldn't get the main display's color space, using generic";
- } else {
- LOG(ERROR) << "Couldn't get any color space";
- }
- }
-
- return g_system_color_space;
-}
-
-// Add a request for full screen mode. Must be called on the main thread.
-void RequestFullScreen(FullScreenMode mode) {
- DCHECK_LT(mode, kNumFullScreenModes);
- if (mode >= kNumFullScreenModes)
- return;
-
- DCHECK_GE(g_full_screen_requests[mode], 0);
- g_full_screen_requests[mode] = std::max(g_full_screen_requests[mode] + 1, 1);
- SetUIMode();
-}
-
-// Release a request for full screen mode. Must be called on the main thread.
-void ReleaseFullScreen(FullScreenMode mode) {
- DCHECK_LT(mode, kNumFullScreenModes);
- if (mode >= kNumFullScreenModes)
- return;
-
- DCHECK_GT(g_full_screen_requests[mode], 0);
- g_full_screen_requests[mode] = std::max(g_full_screen_requests[mode] - 1, 0);
- SetUIMode();
-}
-
-// Switches full screen modes. Releases a request for |from_mode| and adds a
-// new request for |to_mode|. Must be called on the main thread.
-void SwitchFullScreenModes(FullScreenMode from_mode, FullScreenMode to_mode) {
- DCHECK_LT(from_mode, kNumFullScreenModes);
- DCHECK_LT(to_mode, kNumFullScreenModes);
- if (from_mode >= kNumFullScreenModes || to_mode >= kNumFullScreenModes)
- return;
-
- DCHECK_GT(g_full_screen_requests[from_mode], 0);
- DCHECK_GE(g_full_screen_requests[to_mode], 0);
- g_full_screen_requests[from_mode] =
- std::max(g_full_screen_requests[from_mode] - 1, 0);
- g_full_screen_requests[to_mode] =
- std::max(g_full_screen_requests[to_mode] + 1, 1);
- SetUIMode();
-}
-
-void SetCursorVisibility(bool visible) {
- if (visible)
- [NSCursor unhide];
- else
- [NSCursor hide];
-}
-
-bool ShouldWindowsMiniaturizeOnDoubleClick() {
- // We use an undocumented method in Cocoa; if it doesn't exist, default to
- // |true|. If it ever goes away, we can do (using an undocumented pref key):
- // NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults];
- // return ![defaults objectForKey:@"AppleMiniaturizeOnDoubleClick"] ||
- // [defaults boolForKey:@"AppleMiniaturizeOnDoubleClick"];
- BOOL methodImplemented =
- [NSWindow respondsToSelector:@selector(_shouldMiniaturizeOnDoubleClick)];
- DCHECK(methodImplemented);
- return !methodImplemented ||
- [NSWindow performSelector:@selector(_shouldMiniaturizeOnDoubleClick)];
-}
-
-void GrabWindowSnapshot(NSWindow* window,
- std::vector<unsigned char>* png_representation,
- int* width, int* height) {
- // Make sure to grab the "window frame" view so we get current tab +
- // tabstrip.
- NSView* view = [[window contentView] superview];
- NSBitmapImageRep* rep =
- [view bitmapImageRepForCachingDisplayInRect:[view bounds]];
- [view cacheDisplayInRect:[view bounds] toBitmapImageRep:rep];
- NSData* data = [rep representationUsingType:NSPNGFileType properties:nil];
- const unsigned char* buf = static_cast<const unsigned char*>([data bytes]);
- NSUInteger length = [data length];
- if (buf != NULL && length > 0){
- *width = static_cast<int>([rep pixelsWide]);
- *height = static_cast<int>([rep pixelsHigh]);
- png_representation->assign(buf, buf + length);
- DCHECK(png_representation->size() > 0);
- }
-}
-
-void ActivateProcess(pid_t pid) {
- ProcessSerialNumber process;
- OSStatus status = GetProcessForPID(pid, &process);
- if (status == noErr) {
- SetFrontProcess(&process);
- } else {
- LOG(WARNING) << "Unable to get process for pid " << pid;
- }
-}
-
-// Takes a path to an (executable) binary and tries to provide the path to an
-// application bundle containing it. It takes the outermost bundle that it can
-// find (so for "/Foo/Bar.app/.../Baz.app/..." it produces "/Foo/Bar.app").
-// |exec_name| - path to the binary
-// returns - path to the application bundle, or empty on error
-FilePath GetAppBundlePath(const FilePath& exec_name) {
- const char kExt[] = ".app";
- const size_t kExtLength = arraysize(kExt) - 1;
-
- // Split the path into components.
- std::vector<std::string> components;
- exec_name.GetComponents(&components);
-
- // It's an error if we don't get any components.
- if (!components.size())
- return FilePath();
-
- // Don't prepend '/' to the first component.
- std::vector<std::string>::const_iterator it = components.begin();
- std::string bundle_name = *it;
- DCHECK(it->length() > 0);
- // If the first component ends in ".app", we're already done.
- if (it->length() > kExtLength &&
- !it->compare(it->length() - kExtLength, kExtLength, kExt, kExtLength))
- return FilePath(bundle_name);
-
- // The first component may be "/" or "//", etc. Only append '/' if it doesn't
- // already end in '/'.
- if (bundle_name[bundle_name.length() - 1] != '/')
- bundle_name += '/';
-
- // Go through the remaining components.
- for (++it; it != components.end(); ++it) {
- DCHECK(it->length() > 0);
-
- bundle_name += *it;
-
- // If the current component ends in ".app", we're done.
- if (it->length() > kExtLength &&
- !it->compare(it->length() - kExtLength, kExtLength, kExt, kExtLength))
- return FilePath(bundle_name);
-
- // Separate this component from the next one.
- bundle_name += '/';
- }
-
- return FilePath();
-}
-
-bool SetFileBackupExclusion(const FilePath& file_path, bool exclude) {
- NSString* filePath =
- [NSString stringWithUTF8String:file_path.value().c_str()];
-
- // If being asked to exclude something in a tmp directory, just lie and say it
- // was done. TimeMachine will already ignore tmp directories. This keeps the
- // temporary profiles used by unittests from being added to the exclude list.
- // Otherwise, as /Library/Preferences/com.apple.TimeMachine.plist grows the
- // bots slow down due to reading/writing all the temporary profiles used over
- // time.
-
- NSString* tmpDir = NSTemporaryDirectory();
- // Make sure the temp dir is terminated with a slash
- if (tmpDir && ![tmpDir hasSuffix:@"/"])
- tmpDir = [tmpDir stringByAppendingString:@"/"];
- // '/var' is a link to '/private/var', make sure to check both forms.
- NSString* privateTmpDir = nil;
- if ([tmpDir hasPrefix:@"/var/"])
- privateTmpDir = [@"/private" stringByAppendingString:tmpDir];
-
- if ((tmpDir && [filePath hasPrefix:tmpDir]) ||
- (privateTmpDir && [filePath hasPrefix:privateTmpDir]) ||
- [filePath hasPrefix:@"/tmp/"] ||
- [filePath hasPrefix:@"/var/tmp/"] ||
- [filePath hasPrefix:@"/private/tmp/"] ||
- [filePath hasPrefix:@"/private/var/tmp/"]) {
- return true;
- }
-
- NSURL* url = [NSURL fileURLWithPath:filePath];
- // Note that we always set CSBackupSetItemExcluded's excludeByPath param
- // to true. This prevents a problem with toggling the setting: if the file
- // is excluded with excludeByPath set to true then excludeByPath must
- // also be true when un-excluding the file, otherwise the un-excluding
- // will be ignored.
- bool success =
- CSBackupSetItemExcluded((CFURLRef)url, exclude, true) == noErr;
- if (!success)
- LOG(WARNING) << "Failed to set backup excluson for file '"
- << file_path.value().c_str() << "'. Continuing.";
- return success;
-}
-
-CFTypeRef GetValueFromDictionary(CFDictionaryRef dict,
- CFStringRef key,
- CFTypeID expected_type) {
- CFTypeRef value = CFDictionaryGetValue(dict, key);
- if (!value)
- return value;
-
- if (CFGetTypeID(value) != expected_type) {
- ScopedCFTypeRef<CFStringRef> expected_type_ref(
- CFCopyTypeIDDescription(expected_type));
- ScopedCFTypeRef<CFStringRef> actual_type_ref(
- CFCopyTypeIDDescription(CFGetTypeID(value)));
- LOG(WARNING) << "Expected value for key "
- << base::SysCFStringRefToUTF8(key)
- << " to be "
- << base::SysCFStringRefToUTF8(expected_type_ref)
- << " but it was "
- << base::SysCFStringRefToUTF8(actual_type_ref)
- << " instead";
- return NULL;
- }
-
- return value;
-}
-
-void SetProcessName(CFStringRef process_name) {
- if (!process_name || CFStringGetLength(process_name) == 0) {
- NOTREACHED() << "SetProcessName given bad name.";
- return;
- }
-
- if (![NSThread isMainThread]) {
- NOTREACHED() << "Should only set process name from main thread.";
- return;
- }
-
- // Warning: here be dragons! This is SPI reverse-engineered from WebKit's
- // plugin host, and could break at any time (although realistically it's only
- // likely to break in a new major release).
- // When 10.7 is available, check that this still works, and update this
- // comment for 10.8.
-
- // Private CFType used in these LaunchServices calls.
- typedef CFTypeRef PrivateLSASN;
- typedef PrivateLSASN (*LSGetCurrentApplicationASNType)();
- typedef OSStatus (*LSSetApplicationInformationItemType)(int, PrivateLSASN,
- CFStringRef,
- CFStringRef,
- CFDictionaryRef*);
-
- static LSGetCurrentApplicationASNType ls_get_current_application_asn_func =
- NULL;
- static LSSetApplicationInformationItemType
- ls_set_application_information_item_func = NULL;
- static CFStringRef ls_display_name_key = NULL;
-
- static bool did_symbol_lookup = false;
- if (!did_symbol_lookup) {
- did_symbol_lookup = true;
- CFBundleRef launch_services_bundle =
- CFBundleGetBundleWithIdentifier(CFSTR("com.apple.LaunchServices"));
- if (!launch_services_bundle) {
- LOG(ERROR) << "Failed to look up LaunchServices bundle";
- return;
- }
-
- ls_get_current_application_asn_func =
- reinterpret_cast<LSGetCurrentApplicationASNType>(
- CFBundleGetFunctionPointerForName(
- launch_services_bundle, CFSTR("_LSGetCurrentApplicationASN")));
- if (!ls_get_current_application_asn_func)
- LOG(ERROR) << "Could not find _LSGetCurrentApplicationASN";
-
- ls_set_application_information_item_func =
- reinterpret_cast<LSSetApplicationInformationItemType>(
- CFBundleGetFunctionPointerForName(
- launch_services_bundle,
- CFSTR("_LSSetApplicationInformationItem")));
- if (!ls_set_application_information_item_func)
- LOG(ERROR) << "Could not find _LSSetApplicationInformationItem";
-
- CFStringRef* key_pointer = reinterpret_cast<CFStringRef*>(
- CFBundleGetDataPointerForName(launch_services_bundle,
- CFSTR("_kLSDisplayNameKey")));
- ls_display_name_key = key_pointer ? *key_pointer : NULL;
- if (!ls_display_name_key)
- LOG(ERROR) << "Could not find _kLSDisplayNameKey";
-
- // Internally, this call relies on the Mach ports that are started up by the
- // Carbon Process Manager. In debug builds this usually happens due to how
- // the logging layers are started up; but in release, it isn't started in as
- // much of a defined order. So if the symbols had to be loaded, go ahead
- // and force a call to make sure the manager has been initialized and hence
- // the ports are opened.
- ProcessSerialNumber psn;
- GetCurrentProcess(&psn);
- }
- if (!ls_get_current_application_asn_func ||
- !ls_set_application_information_item_func ||
- !ls_display_name_key) {
- return;
- }
-
- PrivateLSASN asn = ls_get_current_application_asn_func();
- // Constant used by WebKit; what exactly it means is unknown.
- const int magic_session_constant = -2;
- OSErr err =
- ls_set_application_information_item_func(magic_session_constant, asn,
- ls_display_name_key,
- process_name,
- NULL /* optional out param */);
- LOG_IF(ERROR, err) << "Call to set process name failed, err " << err;
-}
-
-// Converts a NSImage to a CGImageRef. Normally, the system frameworks can do
-// this fine, especially on 10.6. On 10.5, however, CGImage cannot handle
-// converting a PDF-backed NSImage into a CGImageRef. This function will
-// rasterize the PDF into a bitmap CGImage. The caller is responsible for
-// releasing the return value.
-CGImageRef CopyNSImageToCGImage(NSImage* image) {
- // This is based loosely on http://www.cocoadev.com/index.pl?CGImageRef .
- NSSize size = [image size];
- ScopedCFTypeRef<CGContextRef> context(
- CGBitmapContextCreate(NULL, // Allow CG to allocate memory.
- size.width,
- size.height,
- 8, // bitsPerComponent
- 0, // bytesPerRow - CG will calculate by default.
- [[NSColorSpace genericRGBColorSpace] CGColorSpace],
- kCGBitmapByteOrder32Host |
- kCGImageAlphaPremultipliedFirst));
- if (!context.get())
- return NULL;
-
- [NSGraphicsContext saveGraphicsState];
- [NSGraphicsContext setCurrentContext:
- [NSGraphicsContext graphicsContextWithGraphicsPort:context.get()
- flipped:NO]];
- [image drawInRect:NSMakeRect(0,0, size.width, size.height)
- fromRect:NSZeroRect
- operation:NSCompositeCopy
- fraction:1.0];
- [NSGraphicsContext restoreGraphicsState];
-
- return CGBitmapContextCreateImage(context);
-}
-
-bool CheckLoginItemStatus(bool* is_hidden) {
- ScopedCFTypeRef<LSSharedFileListItemRef> item(GetLoginItemForApp());
- if (!item.get())
- return false;
-
- if (is_hidden)
- *is_hidden = IsHiddenLoginItem(item);
-
- return true;
-}
-
-void AddToLoginItems(bool hide_on_startup) {
- ScopedCFTypeRef<LSSharedFileListItemRef> item(GetLoginItemForApp());
- if (item.get() && (IsHiddenLoginItem(item) == hide_on_startup)) {
- return; // Already is a login item with required hide flag.
- }
-
- ScopedCFTypeRef<LSSharedFileListRef> login_items(LSSharedFileListCreate(
- NULL, kLSSharedFileListSessionLoginItems, NULL));
-
- if (!login_items.get()) {
- LOG(ERROR) << "Couldn't get a Login Items list.";
- return;
- }
-
- // Remove the old item, it has wrong hide flag, we'll create a new one.
- if (item.get()) {
- LSSharedFileListItemRemove(login_items, item);
- }
-
- NSURL* url = [NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]];
-
- BOOL hide = hide_on_startup ? YES : NO;
- NSDictionary* properties =
- [NSDictionary
- dictionaryWithObject:[NSNumber numberWithBool:hide]
- forKey:(NSString*)kLSSharedFileListLoginItemHidden];
-
- ScopedCFTypeRef<LSSharedFileListItemRef> new_item;
- new_item.reset(LSSharedFileListInsertItemURL(
- login_items, kLSSharedFileListItemLast, NULL, NULL,
- reinterpret_cast<CFURLRef>(url),
- reinterpret_cast<CFDictionaryRef>(properties), NULL));
-
- if (!new_item.get()) {
- LOG(ERROR) << "Couldn't insert current app into Login Items list.";
- }
-}
-
-void RemoveFromLoginItems() {
- ScopedCFTypeRef<LSSharedFileListItemRef> item(GetLoginItemForApp());
- if (!item.get())
- return;
-
- ScopedCFTypeRef<LSSharedFileListRef> login_items(LSSharedFileListCreate(
- NULL, kLSSharedFileListSessionLoginItems, NULL));
-
- if (!login_items.get()) {
- LOG(ERROR) << "Couldn't get a Login Items list.";
- return;
- }
-
- LSSharedFileListItemRemove(login_items, item);
-}
-
-bool WasLaunchedAsHiddenLoginItem() {
- if (!WasLaunchedAsLoginItem())
- return false;
-
- ScopedCFTypeRef<LSSharedFileListItemRef> item(GetLoginItemForApp());
- if (!item.get()) {
- LOG(ERROR) << "Process launched at Login but can't access Login Item List.";
- return false;
- }
- return IsHiddenLoginItem(item);
-}
-
-void NSObjectRetain(void* obj) {
- id<NSObject> nsobj = static_cast<id<NSObject> >(obj);
- [nsobj retain];
-}
-
-void NSObjectRelease(void* obj) {
- id<NSObject> nsobj = static_cast<id<NSObject> >(obj);
- [nsobj release];
-}
-
-} // namespace mac_util
« no previous file with comments | « base/mac_util.h ('k') | base/mac_util_unittest.mm » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698