だからここに私の問題があります。 UIImageViewとUIButtonがある画面を作成しようとしています。ユーザーがボタンを押すと、カメラアプリが開き、写真を撮影します。カメラアプリで[写真を使用]を押すと、アプリの画面に戻り、写真は前述のUIImageViewに配置されます。
これまでに起こったことは、「写真を使用」ボタンを押すと、画像がUIImageViewに正しく配置されますが、アプリが次のエラーでクラッシュすることです。
このアプリは、使用法の説明なしでプライバシーに敏感なデータにアクセスしようとしたためクラッシュしました。アプリのInfo.plistには、ユーザーに方法を説明する文字列値を持つNSPhotoLibraryAddUsageDescriptionキーが含まれている必要がありますアプリはこのデータを使用します。
これまでにやったことは:
キー「Privacy-Photo Library Usage Description」に値「$(PRODUCT_NAME)はライブラリを使用して、キャプチャした写真を処理します」を配置しました。 Info.plistファイル内(ソース形式での記述方法も確認し、Apple Developer Documentation)に従って正しいことを確認します。
また、Info.plistファイルにキー「Privacy-Camera Usage Description」と値「$(PRODUCT_NAME)uses Cameras」を配置しました。
「TARGETS->-> Info-> Custom iOS Target Properties」でチェックすると、ステップ1と2で言及した2つのキー/値ペアが存在します。
これまでに私のコードを提供します。
import UIKit
import Vision
import MobileCoreServices
import AVFoundation
import Photos
class ViewController: UIViewController, UIImagePickerControllerDelegate,
UINavigationControllerDelegate {
var newMedia: Bool?
@IBAction func captureImageButtonPressed(_ sender: Any) {
//let imageName : String = "dolphin"
//randomImageView.image = UIImage.init(named:imageName)
if UIImagePickerController.isSourceTypeAvailable(
UIImagePickerControllerSourceType.camera) {
let imagePicker = UIImagePickerController()
imagePicker.delegate = self
imagePicker.sourceType =
UIImagePickerControllerSourceType.camera
imagePicker.mediaTypes = [kUTTypeImage as String]
imagePicker.allowsEditing = false
self.present(imagePicker, animated: true,
completion: nil)
newMedia = true
}
}
@IBAction func classifyButtonPressed(_ sender: UIButton) {
performVisionRequest()
}
@IBOutlet weak var randomImageView: UIImageView!
@IBOutlet weak var classificationLabel: UILabel!
override func viewDidLoad() {
super.viewDidLoad()
}
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
}
func performVisionRequest() {
let start = DispatchTime.now()
let model = Resnet50()
let request = VNImageRequestHandler(cgImage: randomImageView.image!.cgImage!, options: [:])
do {
let m = try VNCoreMLModel(for: model.model)
let coreMLRequest = VNCoreMLRequest(model: m) { (request, error) in
guard let observation = request.results?.first as? VNClassificationObservation else { return }
let stop = DispatchTime.now()
let nanoTime = stop.uptimeNanoseconds - start.uptimeNanoseconds
let timeInterval = Double(nanoTime)
self.classificationLabel.text = "\(observation.identifier) (\(observation.confidence * 100)%) in \(timeInterval) seconds."
}
try request.perform([coreMLRequest])
} catch {
print(error)
}
}
func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any]) {
let mediaType = info[UIImagePickerControllerMediaType] as! NSString
self.dismiss(animated: true, completion: nil)
if mediaType.isEqual(to: kUTTypeImage as String) {
let image = info[UIImagePickerControllerOriginalImage]
as! UIImage
randomImageView.image = image
if (newMedia == true) {
UIImageWriteToSavedPhotosAlbum(image, self,
#selector(ViewController.image(image:didFinishSavingWithError:contextInfo:)), nil)
} else if mediaType.isEqual(to: kUTTypeMovie as String) {
// Code to support video here
}
}
}
@objc func image(image: UIImage, didFinishSavingWithError error: NSErrorPointer, contextInfo:UnsafeRawPointer) {
if error != nil {
let alert = UIAlertController(title: "Save Failed",
message: "Failed to save image",
preferredStyle: UIAlertControllerStyle.alert)
let cancelAction = UIAlertAction(title: "OK",
style: .cancel, handler: nil)
alert.addAction(cancelAction)
self.present(alert, animated: true,
completion: nil)
}
}
func imagePickerControllerDidCancel(_ picker: UIImagePickerController) {
self.dismiss(animated: true, completion: nil)
}
}
上記のエラーを太字で表示する理由はありますか?ご清聴ありがとうございました。
NSPhotoLibraryAddUsageDescriptionはiOS 11で追加されました。
Info.plistに「Privacy-Photo Library Additions Usage Description」を、他のプライバシー権限で行ったように、使用法の説明(文字列)とともに追加してください。