OS X10.11のスタートアップアイテムにアプリケーションを追加する関数を作成する必要があります。それは私が現時点で見つけたものです:
func applicationIsInStartUpItems() -> Bool {
return (itemReferencesInLoginItems().existingReference != nil)
}
func itemReferencesInLoginItems() -> (existingReference: LSSharedFileListItemRef?, lastReference: LSSharedFileListItemRef?) {
if let appUrl : NSURL = NSURL.fileURLWithPath(NSBundle.mainBundle().bundlePath) {
let loginItemsRef = LSSharedFileListCreate(nil, kLSSharedFileListSessionLoginItems.takeRetainedValue(), nil).takeRetainedValue() as LSSharedFileListRef?
if loginItemsRef != nil {
let loginItems: NSArray = LSSharedFileListCopySnapshot(loginItemsRef, nil).takeRetainedValue() as NSArray
if(loginItems.count > 0) {
let lastItemRef: LSSharedFileListItemRef = loginItems.lastObject as! LSSharedFileListItemRef
for var i = 0; i < loginItems.count; ++i {
let currentItemRef: LSSharedFileListItemRef = loginItems.objectAtIndex(i) as! LSSharedFileListItemRef
if let itemURL = LSSharedFileListItemCopyResolvedURL(currentItemRef, 0, nil) {
if (itemURL.takeRetainedValue() as NSURL).isEqual(appUrl) {
return (currentItemRef, lastItemRef)
}
}
}
return (nil, lastItemRef)
} else {
let addatstart: LSSharedFileListItemRef = kLSSharedFileListItemBeforeFirst.takeRetainedValue()
return(nil,addatstart)
}
}
}
return (nil, nil)
}
func toggleLaunchAtStartup() {
let itemReferences = itemReferencesInLoginItems()
let shouldBeToggled = (itemReferences.existingReference == nil)
if let loginItemsRef = LSSharedFileListCreate( nil, kLSSharedFileListSessionLoginItems.takeRetainedValue(), nil).takeRetainedValue() as LSSharedFileListRef? {
if shouldBeToggled {
if let appUrl : CFURLRef = NSURL.fileURLWithPath(NSBundle.mainBundle().bundlePath) {
LSSharedFileListInsertItemURL(loginItemsRef, itemReferences.lastReference, nil, nil, appUrl, nil, nil)
}
} else {
if let itemRef = itemReferences.existingReference {
LSSharedFileListItemRemove(loginItemsRef,itemRef);
}
}
}
}
しかし、LSSharedFileListCreate
、LSSharedFileListInsertItemURL
、LSSharedFileListItemRemove
、kLSSharedFileListItemBeforeFirst
、LSSharedFileListItemCopyResolvedURL
、LSSharedFileListCopySnapshot
、kLSSharedFileListSessionLoginItems
はOS X10.11では非推奨です。これを最新バージョンのMacOSで機能させるにはどうすればよいですか?このコードを変更または書き直す方法は?
Swift 3.0では、次のようになります。
メインアプリケーションのAppDelegate:
func applicationDidFinishLaunching(_ aNotification: Notification) {
// Check if the launcher app is started
var startedAtLogin = false
for app in NSWorkspace.shared().runningApplications {
if app.bundleIdentifier == NCConstants.launcherApplicationIdentifier {
startedAtLogin = true
}
}
// If the app's started, post to the notification center to kill the launcher app
if startedAtLogin {
DistributedNotificationCenter.default().postNotificationName(NCConstants.KILLME, object: Bundle.main.bundleIdentifier, userInfo: nil, options: DistributedNotificationCenter.Options.deliverImmediately)
}
}
LauncherアプリケーションAppDelegateの場合:
func applicationDidFinishLaunching(_ aNotification: Notification) {
let mainAppIdentifier = "<main-app-bundle-id>"
let running = NSWorkspace.shared().runningApplications
var alreadyRunning = false
// loop through running apps - check if the Main application is running
for app in running {
if app.bundleIdentifier == mainAppIdentifier {
alreadyRunning = true
break
}
}
if !alreadyRunning {
// Register for the notification killme
DistributedNotificationCenter.default().addObserver(self, selector: #selector(self.terminate), name: NCConstants.KILLME, object: mainAppIdentifier)
// Get the path of the current app and navigate through them to find the Main Application
let path = Bundle.main.bundlePath as NSString
var components = path.pathComponents
components.removeLast(3)
components.append("MacOS")
components.append("<your-app-name>")
let newPath = NSString.path(withComponents: components)
// Launch the Main application
NSWorkspace.shared().launchApplication(newPath)
}
else {
// Main application is already running
self.terminate()
}
}
func terminate() {
print("Terminate application")
NSApp.terminate(nil)
}
最終的に、メインアプリケーションで、トグルボタン付きのユーザーインターフェイスを追加しました。ユーザーは、ログイン時にアプリを起動するかどうかを選択できます。選択内容はUserDefaultsに保存されます。 View Controllerの場合:
@IBAction func toggleLaunchAtLogin(_ sender: Any) {
if toggleOpenAppLogin.selectedSegment == 0 {
if !SMLoginItemSetEnabled(NCConstants.launcherApplicationIdentifier as CFString, true) {
print("The login item was not successfull")
toggleOpenAppLogin.setSelected(true, forSegment: 1)
}
else {
UserDefaults.standard.set("true", forKey: "appLoginStart")
}
}
else {
if !SMLoginItemSetEnabled(NCConstants.launcherApplicationIdentifier as CFString, false) {
print("The login item was not successfull")
toggleOpenAppLogin.setSelected(true, forSegment: 0)
}
else {
UserDefaults.standard.set("false", forKey: "appLoginStart")
}
}
}
これが誰かに役立つことを願っています。
ここで、サービス管理フレームワークを使用する必要があります。アプリケーションバンドルに追加するヘルパーアプリケーションを作成します。その仕事は、コードを実行してメインアプリケーションを起動することです。あなたのためのいくつかのリソース: