I am building an app about photos and
I want to create a photo sharing feature like Apple's Photos App.
Please see Steps to Reproduce and attached project.
The current share method has the following issues
The file name of the shared photo changes to “FullSizeRender”.
The creation and update dates of shared photos will change to the date they were edited or shared.
I want to ensure that the following conditions are definitely met
Share the latest edited version.
The creation date should be when the original photo was first created.
How can I improve the code?
STEPS TO REPRODUCE
class PHAssetShareManager {
static func shareAssets(_ assets: [PHAsset], from viewController: UIViewController, sourceView: UIView) {
let manager = PHAssetResourceManager.default()
var filesToShare: [URL] = []
let group = DispatchGroup()
for asset in assets {
group.enter()
getAssetFile(asset, resourceManager: manager) { fileURL in
if let fileURL = fileURL {
filesToShare.append(fileURL)
}
group.leave()
}
}
group.notify(queue: .main) {
self.presentShareSheet(filesToShare, from: viewController, sourceView: sourceView)
}
}
private static func getAssetFile(_ asset: PHAsset, resourceManager: PHAssetResourceManager, completion: @escaping (URL?) -> Void) {
print("getAssetFile")
let resources: [PHAssetResource]
switch asset.mediaType {
case .image:
if asset.mediaSubtypes.contains(.photoLive) {
// let editedResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .fullSizePairedVideo }
// let originalResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .pairedVideo }
let editedResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .fullSizePhoto }
let originalResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .photo }
resources = editedResources.isEmpty ? originalResources : editedResources
} else {
let editedResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .fullSizePhoto }
let originalResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .photo }
resources = editedResources.isEmpty ? originalResources : editedResources
}
case .video:
let editedResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .fullSizeVideo }
let originalResources = PHAssetResource.assetResources(for: asset).filter { $0.type == .video }
resources = editedResources.isEmpty ? originalResources : editedResources
default:
print("Unsupported media type")
completion(nil)
return
}
guard let resource = resources.first else {
print("No resource found")
completion(nil)
return
}
let fileName = resource.originalFilename
let tempDirectoryURL = FileManager.default.temporaryDirectory
let localURL = tempDirectoryURL.appendingPathComponent(fileName)
// Delete existing files and reset cache
if FileManager.default.fileExists(atPath: localURL.path) {
do {
try FileManager.default.removeItem(at: localURL)
} catch {
print("Error removing existing file: \(error)")
}
}
let options = PHAssetResourceRequestOptions()
options.isNetworkAccessAllowed = true
resourceManager.writeData(for: resource, toFile: localURL, options: options) { (error) in
if let error = error {
print("Error writing asset data: \(error)")
completion(nil)
} else {
completion(localURL)
}
}
}
private static func presentShareSheet(_ items: [Any], from viewController: UIViewController, sourceView: UIView) {
print("presentShareSheet")
let activityViewController = UIActivityViewController(activityItems: items, applicationActivities: nil)
if UIDevice.current.userInterfaceIdiom == .pad {
activityViewController.popoverPresentationController?.sourceView = sourceView
activityViewController.popoverPresentationController?.sourceRect = sourceView.bounds
}
viewController.present(activityViewController, animated: true, completion: nil)
}
}```
Post
Replies
Boosts
Views
Activity
https://developer.apple.com/documentation/photokit/browsing_and_modifying_photo_albums
I am thinking of making a photo editing application (photo rotation application) by referring to the sample project that can be downloaded from this URL.
In this sample project, when I try to filter photos with sepia or chrome, some photos work and some do not work.
When things don't work, usually run into errors like the one below.
Can't edit the asset: Optional (Error Domain = NSCocoaErrorDomain Code = -1 "(null)")
I don't even know the correlation between photos that succeed and photos that fail to edit.
It's strange that Apple's genuine photo app can rotate all photos.
Is the structure different from the genuine app?
How can I successfully edit (filter and rotate) all kinds of photos based on this sample code?
The code below is the code in the sample project.
It seems that the image is generated using ".ciContext.writeJPEGRepresentation
PHPhotoLibrary.shared().performChanges({
														let request = PHAssetChangeRequest(for: self.asset)
														request.contentEditingOutput = output
												}
Seems to be making changes to the photos stored in the library.
Sample Project
// Returns a filter-applier function for the named filter.
		// Use the function as a handler for a UIAlertAction object.
		/// - Tag: ApplyFilter
		func getFilter(_ filterName: String) -> (UIAlertAction) -> Void {
				func applyFilter(_: UIAlertAction) {
						// Set up a handler to handle prior edits.
						let options = PHContentEditingInputRequestOptions()
						options.canHandleAdjustmentData = {
								$0.formatIdentifier == self.formatIdentifier && $0.formatVersion == self.formatVersion
						}
						
						// Prepare for editing.
						asset.requestContentEditingInput(with: options, completionHandler: { input, info in
								guard let input = input
										else { fatalError("Can't get the content-editing input: \(info)") }
								
								// This handler executes on the main thread; dispatch to a background queue for processing.
								DispatchQueue.global(qos: .userInitiated).async {
										
										// Create adjustment data describing the edit.
										let adjustmentData = PHAdjustmentData(formatIdentifier: self.formatIdentifier,
																													formatVersion: self.formatVersion,
																													data: filterName.data(using: .utf8)!)
										
										// Create content editing output, write the adjustment data.
										let output = PHContentEditingOutput(contentEditingInput: input)
										output.adjustmentData = adjustmentData
										
										// Select a filtering function for the asset's media type.
										let applyFunc: (String, PHContentEditingInput, PHContentEditingOutput, @escaping () -> Void) -> Void
										if self.asset.mediaSubtypes.contains(.photoLive) {
												applyFunc = self.applyLivePhotoFilter
										} else if self.asset.mediaType == .image {
												applyFunc = self.applyPhotoFilter
										} else {
												applyFunc = self.applyVideoFilter
										}
										
										// Apply the filter.
										applyFunc(filterName, input, output, {
												// When the app finishes rendering the filtered result, commit the edit to the photo library.
												PHPhotoLibrary.shared().performChanges({
														let request = PHAssetChangeRequest(for: self.asset)
														request.contentEditingOutput = output
												}, completionHandler: { success, error in
														if !success { print("Can't edit the asset: \(String(describing: error))") }
												})
										})
								}
						})
				}
				return applyFilter
		}
		
		func applyPhotoFilter(_ filterName: String, input: PHContentEditingInput, output: PHContentEditingOutput, completion: () -> Void) {
				
				// Load the full-size image.
				guard let inputImage = CIImage(contentsOf: input.fullSizeImageURL!)
						else { fatalError("Can't load the input image to edit.") }
				
				//MARK:	Apply the filter and Rotation
				let outputImage = inputImage
						.oriented(forExifOrientation: input.fullSizeImageOrientation)
						.applyingFilter(filterName, parameters: [:]).oriented(.right)
				
				// Write the edited image as a JPEG.
				do {
						//MARK: I get errors here
						try self.ciContext.writeJPEGRepresentation(of: outputImage,
																											 to: output.renderedContentURL, colorSpace: inputImage.colorSpace!, options: [:])
				} catch let error {
						fatalError("Can't apply the filter to the image: \(error).")
				}
				completion()
		}
I would appreciate it if you could answer
Thank you in advance.
HelloI switched to a new Mac recently.I have already submitted the app to the App Store and I have the app in the App Store.I want to copy and move the app project from my old Mac to my new Mac.Would something go wrong if I modified the project on a new Mac and submitted the build to the App Store on the new Mac?For example, the data that the user saved in the app is lost.I am so scared that I cannot submit new update build.Can I submit the copied project to the app store?Thank you in advance