Conflict resolution when saving PDFs on iOS
Nutrient iOS SDK offers a built-in mechanism for detecting external file changes by leveraging system support for file presentation callbacks. In some cases, these outside file modifications can conflict with unsaved changes performed on the loaded document. The framework can detect these situations and offers both a bundled UI to present conflict resolution options to your end users and an API that enables you to perform conflict resolution programmatically.
The conflict resolution system builds on top of system file presentation APIs. Before continuing with this guide, you should familiarize yourself with file coordination and file presentation basics and how Nutrient implements those features by reading the file coordination guide.
Conflict resolution options
Nutrient iOS SDK currently offers three conflict resolution options:
- Close — The document is removed from the UI. This is mostly applicable during file deletion.
- Save — The in-memory changes are written to disk, overriding any external modifications. If the file was previously deleted, it gets restored after invoking this option.
- Reload — Any in-memory changes are discarded and the file is reloaded from disk. This is not applicable to externally deleted files.
The above options have corresponding definitions in the FileConflictResolution enum.
Default actions
The framework will, in some cases, automatically pick the most appropriate conflict resolution option by:
- Triggering an automatic reload from disk if the file changed and either there are no unsaved local changes or conflict resolution is not available.
- Removing the document from the containing view controller if the on-disk file was removed and either the controller is currently not visible or conflict resolution is not available.
In all other cases, a conflict resolution UI with user-selectable options is presented.
UI
The built-in conflict resolution UI uses a UIAlertController to present selectable options to end users.
| File changes | File deletion | 
|---|---|
|  |  | 
Customization
The central customization point for conflict resolution is ConflictResolutionManager. Nutrient view controllers — such as PDFViewController, MultiDocumentViewController, and PDFTabbedViewController — are set up to listen for PSPDFDocumentUnderlyingFileChanged, which is forwarded to their internal conflict resolution manager instances. You can subclass ConflictResolutionManager and leverage the provided subclassing hooks to customize the conflict resolution UI or to suppress the UI completely and perform default conflict resolution actions.
After a conflict resolution option is selected, control is returned to the invoking view controller via ConflictResolutionManagerDelegate. The delegate handler then performs required UI actions to honor the selected option and invokes conflict resolution actions on Document via resolveFileConflict(forDataProvider:with:).
Disabling conflict resolution
A ConflictResolutionManager subclass can also be used to completely disable conflict resolution by overriding the PSPDFDocumentUnderlyingFileChanged handler, handleUnderlyingFileChangedNotification(_:), and not calling through to the default implementation. Nutrient should continue to function normally in this case if APFS copies are available for the file in question. Saving could lead to undefined behavior, unless Document.SaveStrategy.rewrite is used.
Conflict resolution actions will also not be performed if file coordination is disabled.
APFS copies
To offer conflict resolution options, Nutrient needs to preserve the full document state as it was before any external modifications took place. Since the framework was designed to handle very large PDF documents, it won’t always have the full file data in memory for larger documents. If memory pressure raises too high, the data will be read incrementally and discarded from memory. This makes conflict resolution more challenging than with simple document-based applications, which can have the full file data available in memory at all times.
To overcome this hurdle, the conflict resolution system in Nutrient relies on lightweight APFS(opens in a new tab)-powered file copies. By default, the framework will create an efficient temporary file copy when a file is first loaded via CoordinatedFileDataProvider — the default data provider used by Document. Due to the copy-on-write behavior supported by APFS(opens in a new tab), the copies are created with minimal overhead and do not cause any additional disk space use. Nutrient keeps monitoring the original file for changes, but at the same time, uses its safe internal copy for more efficient file access and to support conflict resolution actions should file conflicts arise.
Since conflict resolution relies on APFS(opens in a new tab), it is only available for files residing on volumes that are using this file system. APFS(opens in a new tab) is the default file system for iOS devices running iOS 10.3 and later. On the desktop, APFS(opens in a new tab) is available on macOS High Sierra and later. You can determine whether or not conflict resolution is currently available by checking the isConflictResolutionAvailable flag on CoordinatedFileDataProvider.
 
  
  
  
 