iOS Swift: Separate AppDelegate for XCTest

前端 未结 3 808
感动是毒
感动是毒 2021-02-19 04:27

Due to a couple issues, I want the XCTest target in a project to run a separate app delegate. Using ObjC, this was a relatively straightforward process: manipulate main.m<

相关标签:
3条回答
  • 2021-02-19 04:57

    This solution to this is as follows:

    1. Duplicate your existing application's Target and rename it to something appropriate. In your case maybe 'TestingHarness' or some such. Note that you'll also want to change the bundle identifier and rename the corresponding Info.plist file. Renaming the Info.plist file means you'll need to change the Info.plist filename setting in your new target's Build Settings tab to match the new name.

    2. Create another *AppDelegate.swift file. In your case I'd call it TestAppDelegate.swift.

    3. Copy over your existing AppDelegate.swift file's contents into TestAppDelegate.swift and edit as desired. Make sure to leave the @UIApplicationMain annotation and implement the needed UIApplicationDelegate callbacks.

    4. Change the target membership of each of your *AppDelegate.swift files so that AppDelegate.swift is not included in your new 'TestHarness' target and TestAppDelegate.swift is not included in your main app's target. (You edit a file's Target Membership by selecting it in the File Browser and opening the File Inspector which you can access in the right-sidebar by default, or by choosing it in the menu under View -> Utilities.)

    5. Now you have two separate targets with separate App Delegates that you can build and run independently. The final step is to select your new 'TestHarness' target as the Host Application for your test targets. (Click the top-level project entry in the File Browser, then click your desired test Target in the sub-listing. On the General tab you'll see Host Application as the only available drop down.)

    Note: these instructions are for Xcode 7.2.

    0 讨论(0)
  • 2021-02-19 05:13

    To achieve this is Swift you need to take a couple of steps:

    1. If you are using Storyboards, create your view stack programmatically on your AppDelegate.

      Remove Main.storyboard from your project configuration

      Delete @UIApplicationMain from the beginning of your AppDelegate and add this code.

      func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
         let storyboard = UIStoryboard(name: "Main", bundle: nil)
         let vc = storyboard.instantiateInitialViewController()
      
         let window = UIWindow(frame: UIScreen.main.bounds)
         window.rootViewController = vc
      
         window.makeKeyAndVisible()
         self.window = window
      
         return true
      }
      
    2. Create a new file at the root your target and call it main.swift.

      Add this code if you don't need to do any setup for your tests

      import UIKit
      
      let kIsRunningTests = NSClassFromString("XCTestCase") != nil
      let kAppDelegateClass = kIsRunningTests ? nil : NSStringFromClass(AppDelegate.self)
      
      UIApplicationMain(CommandLine.argc, CommandLine.unsafeArgv, nil, kAppDelegateClass)
      

      If you need to make some configuration before you run the tests, create a new class FakeAppDelegate as a subclass from NSObject and add your setup code there.

      Put this code in main.swift

      import UIKit
      
      let kIsRunningTests = NSClassFromString("XCTestCase") != nil
      let kAppDelegateClass = kIsRunningTests ? NSStringFromClass(FakeAppDelegate.self) : NSStringFromClass(AppDelegate.self)
      
      UIApplicationMain(CommandLine.argc, CommandLine.unsafeArgv, nil, kAppDelegateClass)
      
    0 讨论(0)
  • 2021-02-19 05:17

    It's strongly unrecommended to add conditions to normal code checking if its being tested. Instead you should mock your AppDelegate in tests to do whatever you want.

    Then you could replace delegate of UIApplication is setUp in super class of your each XCTestCase'es.

    class MockAppDelegate:NSObject, UIApplicationDelegate {
    
    }
    
    
    class BaseTest: XCTestCase {
    
        override func setUp() {
            super.setUp()
            UIApplication.shared.delegate = MockAppDelegate()
           }
    }
    class Test1: BaseTest {
    
        override func setUp() {
            super.setUp()
            // normal testing
           }
    }
    

    If you still want to stop code execution for tests this is my method that works well:

    You can add startup parameter to app which indicates that this is test run App Start execution

    These parameters are accessible from NSUserDefaults

    #define IS_TESTS [[NSUserDefaults standardUserDefaults] boolForKey:@"TESTING"]
    
    0 讨论(0)
提交回复
热议问题