我有一个项目,其中有许多使用 imagePicker 的视图控制器。每次,我都必须再次复制 didFinishPickingMediaWithInfo 并且只更改一些代码。
然后我决定把 UIImagePickerControllerDelegate 和 UINavigationControllerDelegate 包装成我自己的协议,并扩展这个协议来实现 didFinishPickingMediaWithInfo。但是,根本没有调用 didFinishPickingMediaWithInfo。所有其他部分都运行良好,图像拾取器和相机视图显示良好,但完成拾取后,没有调用 didFinish 函数。
我在网上看到了一些建议,比如这个。他们使用具体类来扭曲这两个协议,而不是接口。
https://gist.github.com/rpassis/4622291029cd12e4ce2b7585d3e62d15
我不知道为什么我的解决方案是错误的,有人可以告诉我我的代码错误的原因。也许我误解了协议和协议扩展的某些部分。顺便说一句,我发现一个警告是
非'@objc' 方法'imagePickerController(_:didFinishPickingMediaWithInfo:)' 不满足'@objc' 协议'UIImagePickerControllerDelegate' 的可选要求
另一种解释来自 Swift 协议实现另一个 @objc 协议
我的代码如下所示。
public protocol ImagePickerDelegate: UIImagePickerControllerDelegate, UINavigationControllerDelegate {
func successActionFromCamera(with localIdentifier: String, picker: UIImagePickerController)
func successActionFromPhotoLibrary(with imageURL: URL, picker: UIImagePickerController)
}
扩展我的自定义委托
extension ImagePickerDelegate {
public func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any]) {
//camera
if info[UIImagePickerControllerReferenceURL] == nil {
func savePhotoAndTakeAction() {
var imagePlaceholder:PHObjectPlaceholder!
DispatchQueue.global(qos: .default).async {
PHPhotoLibrary.shared().performChanges({
let request = PHAssetChangeRequest.creationRequestForAsset(from: info[UIImagePickerControllerOriginalImage]! as! UIImage)
imagePlaceholder = request.placeholderForCreatedAsset!
}, completionHandler: { (success, error) -> Void in
DispatchQueue.main.async {
if success {
//image saved to photos library.
self.successActionFromCamera(with: imagePlaceholder.localIdentifier, picker: picker)
} else {
picker.dismiss(animated: true, completion: nil)
print(error!.localizedDescription)
}
picker.dismiss(animated: true, completion: nil)
}
})
}
}
switch PHPhotoLibrary.authorizationStatus() {
case .denied:
picker.dismiss(animated: false) { BasePhotoUtil.showAccessAlertController(false) }
return
case .notDetermined:
PHPhotoLibrary.requestAuthorization({ (newStatus) in
if (newStatus == .authorized) {
savePhotoAndTakeAction()
}
else {
DispatchQueue.main.async {
picker.dismiss(animated: false, completion: { BasePhotoUtil.showAccessAlertController(false) })
}
return
}
})
default:
break
}
savePhotoAndTakeAction()
} else {
//photo library
if let imageURL = info[UIImagePickerControllerReferenceURL] as? URL {
self.successActionFromPhotoLibrary(with: imageURL, picker: picker)
} else {
picker.dismiss(animated: true, completion: nil)
}
}
}
呈现 ImagePicker 的函数
private static func showImagePickerView(isCamera: Bool, currentVC: UIViewController) {
let imagePicker = UIImagePickerController()
imagePicker.delegate = currentVC as? UIImagePickerControllerDelegate & UINavigationControllerDelegate
imagePicker.allowsEditing = false
imagePicker.navigationBar.isTranslucent = false
if isCamera {
if UIImagePickerController.isSourceTypeAvailable(.camera) {
imagePicker.sourceType = .camera
imagePicker.cameraCaptureMode = .photo
} else {
BaseAlertUtil.showNoFunctionAlertController(title: "No Camera", message: "Sorry, this device has no camera")
}
} else {
imagePicker.sourceType = .photoLibrary
}
currentVC.present(imagePicker, animated: true) {
BaseThemeUtil.setStatusBarStyle(.default)
}
}