Инициализатор недоступен из-за «внутреннего» уровня защиты

89

У меня есть протоколы

ВойтиСтратегия

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
}

и два класса:

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
    }

}

FacebookВходСтратегии

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)
    }
}

В моем ViewController :

Когда я использую:

let facebookLoginProvider = LoginProvider(strategy: FacebookLoginStrategy())

Он говорит:

FacebookLoginStrategy недоступен из-за «внутреннего» уровня защиты

appiconhero.co
источник

Ответы:

250

Просто добавьте в свою FacebookLoginStrategy:

public init() {}

Пока вы не реализуете init () явно, он по умолчанию помечен как внутренний. Вам необходимо перезаписать этот уровень разрешений, чтобы иметь возможность создавать экземпляры вне вашей структуры.

jboi
источник
13
Отличный ответ! И следите за Swift4, на самом деле это ошибка, если вы его не реализуете.
Пол Разван Берг
2
Просто в класс.
jboi
12
это настолько глупо, что этого требует Swift. Если я объявлю весь тип класса общедоступным, бесплатный init () также станет общедоступным
LightningStryk
3
Это должно быть ошибкой Swift. Она должна не ошибка компилятора, так как мы ознаменовали publicдля class/struct.
lee
1
Я добавил public перед своей инициализацией и все еще получаю то же сообщение. Есть еще защита?
AndreG
8

Если вы запускаете это в коде в XCTestCase, убедитесь, что вы добавили его @testable import My-Awesome-Appв начало тестового файла.

Алекс Заватоне
источник