iosswiftuidocumentinteractionuidocumentuidocumentpickerviewcontroller

swift iOS: create UIDocument in app documents folder


i am looking for a way how to do it. I say option how to work with files which are alredy created but UIDocument can not initialized only from URL.

So, here is my code and it does not work:

        weak var weakSelf = self
        let toFolder = presenter.interactor.url.path
        let name = "Untitled"
        var toPath = toFolder + "/" + name
        var count = 1
        while FileManager.default.fileExists(atPath: toPath) {
            toPath = toFolder + "/" + name + " (\(count))"
            count += 1
        }
        let url = (toPath + ".UTI").url!
        print(url.absoluteString)
        let document = Document(fileURL: url)
        document.save(to: url, for: UIDocument.SaveOperation.forCreating, completionHandler: { (success) in
            if success {
                vc.document = document
                vc.title = url.lastPathComponent
                let nvc = UINavigationController(rootViewController: vc)
                weakSelf?.userInterface.present(nvc, animated: true, completion: nil)
            } else {
                Alert.showError(body: "Could not create document".localized)
            }
        })

url is:

/var/mobile/Containers/Data/Application/67C2A474-A054-4DFE-8587-B453A8B44554/Documents/Untitled.UTI

I have crash on string: "let document = Document(fileURL: url)"

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'must pass a valid file URL to -[UIDocument initWithFileURL:]'

class Document:

enum DocumentError: Error {
case saveError
case loadError
}

class Document: UIDocument {

var text: String?

var encodingUsed: String.Encoding?

override func contents(forType typeName: String) throws -> Any {

    guard let data = text?.data(using: .utf8) else {
        throw DocumentError.saveError
    }

    return data
}

override func load(fromContents contents: Any, ofType typeName: String?) throws {

    guard let data = contents as? Data else {
        throw DocumentError.loadError
    }

    guard let utf8 = String(data: data, encoding: .utf8) else {
        throw DocumentError.loadError
    }

    self.text = utf8

}

}

var url: URL? {
    if self != "" && !isEmpty {
        return URL(string: self)
    } else {
        return nil
    }
}

Solution

  • Your extension should return:URL(fileURLWithPath: self) not URL(string: self) because this is local file.