Mac OS X: start on launch while in app store?

前端 未结 4 1480
不知归路
不知归路 2020-12-23 23:51

I need to start my app at system launch, but the problem is: it is already in App Store, so I must follow some rules like using sandboxing. This leads to fail of desired fun

相关标签:
4条回答
  • 2020-12-24 00:22

    For Sandboxed applications you need to create special Login Item helper application (located inside Contents/Library/LoginItems). See more here. Also note that your app must be launched from /Applications folder for correct work of Login Item.

    0 讨论(0)
  • 2020-12-24 00:34

    I recently went through this same process, unfortunately with sandboxing it's not nearly as easy to do as it used to be. I made a test application with very detailed instructions that is now on Github

    Notes

    This demo application and your application will only work if they are deployed preferably in the /Applications/MyGreat.app and will not work reliably from the Xcode debug folder.

    Project settings

    These are the settings of my project that worked perfectly with this implementation.

    1. Create a new project with ARC enabled
    2. Sandbox both your main application and helper application (if you haven't created a helper yet we will get to it shortly) I also enabled code signing
    3. Since this was just a test application I had no active entitlements for either the main application or the helper
    4. If you haven't already, create a helper app. Go to your project settings and click "Add Target" choose a Cocoa Application. Name it something such as MyAwesomeProjectHelper also with ARC enabled. (I left its "App Store Category" blank)
    5. Now select the Target of your main application. Go to Build Phases -> Add Build Phase -> Add Copy Files.
    6. Change the Destination to Wrapper. Make the Subpath Contents/Library/LoginItems leave Copy only when installing unchecked. Drag your helper application from Products on the left into the tableview.

    Main Application code setup

    1. Import ServiceManagement.framework into your main application(not your helper) and include #import <ServiceManagement/ServiceManagement.h> in your .h file
    2. Grab StartAtLoginController from Github. This is an easy to use class by Alex Zielenski to deal with the complications of adding, removing and querying login items. Import StartAtLoginController.h into your h file.
    3. Create whatever interface you want to for controlling this setting. If your application automatically enables this it will be denied from the Mac App Store(per guideline #2.26)
    4. Implement a method such as - (IBAction)checkChanged:(id)sender I made a simple checkbox tied to the StandardUserDefaults. (If you chose to do something else your implementation for this may vary.) I also bound the checkbox to IBOutlet NSButton *loginCheck; in order to determine it's state. This could also be done through [[NSUserDefaults standardUserDefaults] boolForKey:YourKey]
    5. Implement code similar to this in your .m file.

      StartAtLoginController *loginController = [[StartAtLoginController alloc] init];
      [loginController setBundle:[NSBundle bundleWithPath:[[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent:@"Contents/Library/LoginItems/HelperApp.app"]]]; 
      // Change "HelperApp.app" to the name of your helper
      
      if ([loginCheck state]) {
          if (![loginController startAtLogin]) {
              [loginController setStartAtLogin: YES];
          }
      } else {
          if ([loginController startAtLogin]) {
              [loginController setStartAtLogin:NO];
          }
      }
      
    6. That's it. As you can see in this project there are some other methods you may want to use such as:

      if ([loginController startAtLogin]) {
          NSLog(@"Error");
      }
      

      For checking after you enable or disable the setting to make sure it worked correctly. Or this:

      BOOL startsAtLogin = [loginController startAtLogin];
      if (startsAtLogin) {
          // Do stuff
      }
      

      To do something if the login helper is enabled.

    Helper Application code setup

    Make sure to test this code vigorously with your implementation.

    1. Make your helper application a UIElement by navigating to HelperApp.plist located in the Supporting Files group by default. Add a line at the bottom with the Key Application is agent (UIElement) and YES as the Value (this will suppress the application from flashing a dock icon each time the user enables launch at login) I also deleted everything except for the App Delegate in interface builder
    2. Erase the default method - (void)applicationDidFinishLaunching:(NSNotification *)aNotification and replace it with - (void)applicationWillFinishLaunching:(NSNotification *)aNotification
    3. Within this method implement code similar to this.

      NSString *appPath = [[[[[[NSBundle mainBundle] bundlePath] stringByDeletingLastPathComponent] stringByDeletingLastPathComponent] stringByDeletingLastPathComponent] stringByDeletingLastPathComponent];
      // This string takes you from MyGreat.App/Contents/Library/LoginItems/MyHelper.app to MyGreat.App This is an obnoxious but dynamic way to do this since that specific Subpath is required
      NSString *binaryPath = [[NSBundle bundleWithPath:appPath] executablePath]; // This gets the binary executable within your main application
      [[NSWorkspace sharedWorkspace] launchApplication:binaryPath];
      [NSApp terminate:nil];
      

      This code finds your main application, determines it's binary executable(required to launch the application within the sandbox) opens your application, then quits

    4. That's it.

    Deploy

    The last thing you should do when deploying your application for yourself or to the Mac App store is remove your Helper app from the Archived items. Do this by navigating to the Target of your HelperApp -> Build Settings -> Skip Install and set Yes for Release. Apple provides more information at (http://developer.apple.com/library/ios/#documentation/ToolsLanguages/Conceptual/Xcode4UserGuide/000-About_Xcode/about.html)

    0 讨论(0)
  • 2020-12-24 00:35

    The setup for getting your sandboxed app to launch at login is time-consuming and it's too easy to get a step wrong. That's why I made a Swift package to automate it. With my LaunchAtLogin package, all you have to do is add a build step in Xcode and then write two lines of code:

    import LaunchAtLogin
    
    LaunchAtLogin.isEnabled = true
    
    0 讨论(0)
  • 2020-12-24 00:37

    I just implemented a LoginItem for my own app and found information at these links useful:

    How to create a helper application for Mac App to start it on user login?

    http://developer.apple.com/library/mac/#documentation/Security/Conceptual/AppSandboxDesignGuide/DesigningYourSandbox/DesigningYourSandbox.html

    https://github.com/tcurdt/TCLoginItemHelper

    http://www.delitestudio.com/2011/10/25/start-dockless-apps-at-login-with-app-sandbox-enabled/

    0 讨论(0)
提交回复
热议问题