NSFilePresenter プロトコルを採用する UIViewController があります。iCloud のどこにでもあるコンテナに保存されているファイルの変更を受け取るために、presentedItemDidChange() を実装しました。私の実装は iOS13.7 では問題なく動作していますが、iOS 14 では失敗します。
macOS の「Finder」などの外部プロセスから提示されたファイルを変更すると、presentedItemDidChange() は呼び出されません。
これが iOS 14 の既知のバグである場合、または実装に何か不足していますか?
私の実装はおおよそ次のようになります。
var m_basename = "textfile.txt"
var iCloudURL: URL? {
get {
guard let iCloudDocumentsBaseURL = FileManager.default.url(forUbiquityContainerIdentifier: iCloudContainerId) else {
return nil
}
let iCloudDocumentsURL = iCloudDocumentsBaseURL.appendingPathComponent("Documents").appendingPathComponent(m_basename)
return iCloudDocumentsURL
}
}
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
NSFileCoordinator.addFilePresenter(self)
/* do some iCloud availability check here ; not shown */
save(text: "Test string") { (success) in
if success {
print(#function, "Wrote file successfully")
} else {
print(#function, "Write file failed")
}
}
}
public func save(text: String, completion: @escaping (Bool) -> Void) {
if let presenterurl = self.presentedItemURL {
var errorMain: NSError?
// set presenter if we load a file that is currently being presented
var filePresenter: NSFilePresenter? = nil
if presenterurl == iCloudURL {
filePresenter = fPresenter
}
let coord = NSFileCoordinator(filePresenter: filePresenter)
coord.coordinate(writingItemAt: activeURL, options: .forReplacing, error: &errorMain) { (writeUrl) in
do {
try text.write(toFile: writeUrl.path, atomically: true, encoding: String.Encoding.utf8)
print(#function, "\(writeUrl) write OK")
completion(true)
} catch {
print(#function, "\(writeUrl) write failed: \(error.localizedDescription)")
completion(false)
}
}
if errorMain != nil {
print("Error: ", errorMain!.localizedDescription)
completion(false)
}
}
print(#function, "end of save() function")
}
// MARK: - NSFilePresenter
extension ViewController: NSFilePresenter {
public var presentedItemURL: URL? {
return iCloudURL
}
public var presentedItemOperationQueue: OperationQueue {
return OperationQueue.main
}
public func presentedItemDidChange() {
print(#function, "Loading file \(m_basename)")
}
}
ありがとう