Зарегистрироваться в какао как элемент входа?
Google дал мне: http://developer.apple.com/samplecode/LoginItemsAE/index.html
И я подумал, что должен быть лучший способ, чем использование AppleScript Events.
Поэтому я скачал источники Growl. Они используют точные источники из этой статьи разработчиков Apple.
Есть ли способ лучше?
(Я имею в виду пункты "Вход в систему" в разделе "Учетные записи" в "Системных настройках", т. Е. Запуск моей программы при входе пользователя программным способом)
7 ответов
В Leopard есть новый API под названием LSSharedFileList. Одна из вещей, которую он позволяет вам делать, это просматривать и редактировать список элементов входа (называемых элементами входа в сеанс в этом API).
Кстати, я ведущий разработчик Growl. Мы еще не отказались от AE, потому что нам все еще нужен Tiger, но я думаю о том, чтобы сбросить его до 1.2 (пока не обсуждал это с другими разработчиками). Когда мы удаляем Tiger, мы также удаляем LoginItemsAE и переключаемся на API Списка общих файлов.
РЕДАКТИРОВАТЬ с 2012 года: с 2009 года, когда я первоначально написал этот ответ, Growl перешел на LSSharedFileList, и я покинул проект.
Я наткнулся на https://github.com/Mozketo/LaunchAtLoginController Бена Кларка-Робинсона. Очень элегантное решение очень распространенной проблемы.
Это работает на xcode 5.
- (BOOL)isLaunchAtStartup {
// See if the app is currently in LoginItems.
LSSharedFileListItemRef itemRef = [self itemRefInLoginItems];
// Store away that boolean.
BOOL isInList = itemRef != nil;
// Release the reference if it exists.
if (itemRef != nil) CFRelease(itemRef);
return isInList;
}
- (void)toggleLaunchAtStartup {
// Toggle the state.
BOOL shouldBeToggled = ![self isLaunchAtStartup];
// Get the LoginItems list.
LSSharedFileListRef loginItemsRef = LSSharedFileListCreate(NULL, kLSSharedFileListSessionLoginItems, NULL);
if (loginItemsRef == nil) return;
if (shouldBeToggled) {
// Add the app to the LoginItems list.
CFURLRef appUrl = (__bridge CFURLRef)[NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]];
LSSharedFileListItemRef itemRef = LSSharedFileListInsertItemURL(loginItemsRef, kLSSharedFileListItemLast, NULL, NULL, appUrl, NULL, NULL);
if (itemRef) CFRelease(itemRef);
}
else {
// Remove the app from the LoginItems list.
LSSharedFileListItemRef itemRef = [self itemRefInLoginItems];
LSSharedFileListItemRemove(loginItemsRef,itemRef);
if (itemRef != nil) CFRelease(itemRef);
}
CFRelease(loginItemsRef);
}
- (LSSharedFileListItemRef)itemRefInLoginItems {
LSSharedFileListItemRef res = nil;
// Get the app's URL.
NSURL *bundleURL = [NSURL fileURLWithPath:[[NSBundle mainBundle] bundlePath]];
// Get the LoginItems list.
LSSharedFileListRef loginItemsRef = LSSharedFileListCreate(NULL, kLSSharedFileListSessionLoginItems, NULL);
if (loginItemsRef == nil) return nil;
// Iterate over the LoginItems.
NSArray *loginItems = (__bridge NSArray *)LSSharedFileListCopySnapshot(loginItemsRef, nil);
for (id item in loginItems) {
LSSharedFileListItemRef itemRef = (__bridge LSSharedFileListItemRef)(item);
CFURLRef itemURLRef;
if (LSSharedFileListItemResolve(itemRef, 0, &itemURLRef, NULL) == noErr) {
// Again, use toll-free bridging.
NSURL *itemURL = (__bridge NSURL *)itemURLRef;
if ([itemURL isEqual:bundleURL]) {
res = itemRef;
break;
}
}
}
// Retain the LoginItem reference.
if (res != nil) CFRetain(res);
CFRelease(loginItemsRef);
CFRelease((__bridge CFTypeRef)(loginItems));
return res;
}
Я делаю это в приложении, которое я пишу:
Проверьте UKLoginItemRegistry для простого способа сделать это прагматично. Афайк, в Tiger нет способа сделать это без Apple Events; в Leopard есть лучший способ, но если вы используете UKLoginItemRegistry, это действительно не проблема. Вот полный код для реализации пункта меню "Открыть при входе в систему"
+ (bool)isAppSetToRunAtLogon {
int ret = [UKLoginItemRegistry indexForLoginItemWithPath:[[NSBundle mainBundle] bundlePath]];
NSLog(@"login item index = %i", ret);
return (ret >= 0);
}
- (IBAction)toggleOpenAtLogon:(id)sender {
if ([PopupController isAppSetToRunAtLogon]) {
[UKLoginItemRegistry removeLoginItemWithPath:[[NSBundle mainBundle] bundlePath]];
} else {
[UKLoginItemRegistry addLoginItemWithPath:[[NSBundle mainBundle] bundlePath] hideIt: NO];
}
}
Я реорганизовал некоторые ответы здесь, чтобы предоставить категорию на NSApplication
это обеспечивает launchAtLogin
имущество.
SMLoginItemSetEnabled - еще один современный вариант, см. Статью
Посмотрите здесь пример с открытым исходным кодом: https://github.com/invariant/rhpnotifier (LoginItem.m, LoginItem.h)