由于"内部"保护级别,初始化程序无法访问

Khu*_*ong 63 frameworks access-modifiers initializer ios swift

我有一些协议

LoginStrategy

public protocol LoginStrategy {
    func login(_ viewController: UIViewController)
    func getUserInfo(withCompletionHandler completionHandler: @escaping (_ userInfo: [String: Any]?) -> ())
    func createLoginButton(_ frame: CGRect, withCompletionHandler completionHandler: @escaping (_ loginButton: UIView) -> ())
    func getUserId() -> String
}
Run Code Online (Sandbox Code Playgroud)

和两个类:

LoginProvider

public class LoginProvider {

    public let strategy: LoginStrategy

    public func login(_ viewController: UIViewController) {
        return self.strategy.login(viewController)
    }

    public func getUserInfo(withCompletionHandler completionHandler: @escaping (_ userInfo: [String: Any]?) -> ()) {
        return self.strategy.getUserInfo(withCompletionHandler: completionHandler)
    }

    public func createLoginButton(_ frame: CGRect, withCompletionHandler completionHandler: @escaping (_ loginButton: UIView) -> ()) {
        return self.strategy.createLoginButton(frame, withCompletionHandler: completionHandler)
    }

    public func getUserId() -> String {
        return self.strategy.getUserId()
    }

    public init(strategy: LoginStrategy) {
        self.strategy = strategy
    }

}
Run Code Online (Sandbox Code Playgroud)

FacebookLoginStrategy

import Foundation
import FacebookCore
import FacebookLogin

public class FacebookLoginStrategy: LoginStrategy {

    public var grantedPermissions: Set<Permission>? = nil

    public var declinedPermissions: Set<Permission>? = nil

    public var userId: String = ""

    public func login(_ viewController: UIViewController) {
        let loginManager = LoginManager()
        let permissions: [ReadPermission] = [.publicProfile, .userFriends, .email]
        loginManager.logIn(permissions, viewController: viewController) { loginResult in
            switch loginResult {
            case .failed(let error):
                print(error)
            case .cancelled:
                print("User cancelled login.")
            case .success(let grantedPermissions, let declinedPermissions, let accessToken):
                self.userId = accessToken.userId ?? ""
                self.grantedPermissions = grantedPermissions
                self.declinedPermissions = declinedPermissions
                print("Logged in!")
            }
        }
    }

    public func getUserId() -> String {
        return userId
    }

    public func getUserInfo(withCompletionHandler completionHandler: @escaping (_ userInfo: [String: Any]?) -> ()) {
        let request = GraphRequest(graphPath: "me", parameters: ["fields":"email, name"], accessToken: AccessToken.current, httpMethod: .GET, apiVersion: FacebookCore.GraphAPIVersion.defaultVersion)
        request.start { (response, result) in
            switch result {
            case .success(let value):
                print(value.dictionaryValue)
                completionHandler(value.dictionaryValue)
            case .failed(let error):
                print(error)
            }
        }
    }

    public func createLoginButton(_ frame: CGRect, withCompletionHandler completionHandler: @escaping (_ loginButton: UIView) -> ()) {
        let permissions: [ReadPermission] = [.publicProfile, .userFriends, .email]
        let loginButton = LoginButton(readPermissions: permissions)
        loginButton.frame = frame
        completionHandler(loginButton)
    }
}
Run Code Online (Sandbox Code Playgroud)

在我的ViewController中:

我用的时候:

let facebookLoginProvider = LoginProvider(strategy: FacebookLoginStrategy())
Run Code Online (Sandbox Code Playgroud)

它说:

由于"内部"保护级别,"FacebookLoginStrategy"无法访问

jbo*_*boi 186

只需添加到您的FacebookLoginStrategy:

public init() {}
Run Code Online (Sandbox Code Playgroud)

只要您没有显式实现init(),默认情况下它就会标记为内部.您需要覆盖该权限级别才能从框架外部实例化.

  • 这太愚蠢了,Swift 需要这个。如果我将整个类类型声明为 public,则免费的 init() 也应该是 public (25认同)
  • 很棒的答案!并注意Swift4,如果你不实现它,它实际上是一个错误. (11认同)
  • 这应该是“Swift”的一个错误。它不应该是编译器的错误,因为我们已经为“class/struct”标记了“public”。 (3认同)
  • 刚进班。 (2认同)

Ale*_*one 13

如果您在 XCTestCase 中的代码中遇到此问题,请确保您已将其添加@testable import My-Awesome-App到测试文件的顶部。


Yog*_*ore 8

添加 init 方法作为公共访问

public init() {}
Run Code Online (Sandbox Code Playgroud)