From cd22c68c74312932aae5eb675d7c1ee83610486e Mon Sep 17 00:00:00 2001 From: BAHADUR ZAMAN Date: Fri, 29 Nov 2024 17:32:11 +0500 Subject: [PATCH 1/4] =?UTF-8?q?fix(iOS):=20=F0=9F=90=9B=20Image=20Picker?= =?UTF-8?q?=20to=20Handle=20Partial=20Failures=20Gracefully?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit ## Problem Currently, the iOS image picker fails entirely if any single image fails to process, causing a poor user experience. Additionally, there's a crash when attempting to access the path property of an NSDictionary, resulting in app termination. ## Solution Updated the PHPicker implementation to: - Handle each image independently, allowing partial successes - Prevent crashes by implementing proper type checking and error handling - Improve temporary file management - Add detailed error logging for debugging Key changes: 1. Removed all-or-nothing approach for multiple image selection 2. Added proper error collection without failing the entire operation 3. Implemented better memory management with @autoreleasepool 4. Added unique filename generation to prevent conflicts 5. Improved error messages for debugging 6. Added proper null/type checking for URL handling ## Testing Tested scenarios: - Selecting multiple images (mix of local and iCloud images) - Selecting single image - Canceling selection - Handling invalid file types - Memory usage with large selections - iCloud asset download failures ## Impact Users can now select multiple images and get results even if some images fail to process, rather than losing all selected images due to a single failure. --- ios/Classes/FilePickerPlugin.m | 226 +++++++++++++++++---------------- 1 file changed, 116 insertions(+), 110 deletions(-) diff --git a/ios/Classes/FilePickerPlugin.m b/ios/Classes/FilePickerPlugin.m index 6913ae34..5023f216 100644 --- a/ios/Classes/FilePickerPlugin.m +++ b/ios/Classes/FilePickerPlugin.m @@ -450,13 +450,13 @@ - (void)imagePickerController:(UIImagePickerController *)picker didFinishPicking NSURL *pickedVideoUrl = [info objectForKey:UIImagePickerControllerMediaURL]; NSURL *pickedImageUrl; - + if(@available(iOS 13.0, *)) { - + if(pickedVideoUrl != nil) { NSString * fileName = [pickedVideoUrl lastPathComponent]; NSURL * destination = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:fileName]]; - + if([[NSFileManager defaultManager] isReadableFileAtPath: [pickedVideoUrl path]]) { Log(@"Caching video file for iOS 13 or above..."); [[NSFileManager defaultManager] copyItemAtURL:pickedVideoUrl toURL:destination error:nil]; @@ -465,13 +465,13 @@ - (void)imagePickerController:(UIImagePickerController *)picker didFinishPicking } else { pickedImageUrl = [info objectForKey:UIImagePickerControllerImageURL]; } - + } else { pickedImageUrl = [info objectForKey:UIImagePickerControllerImageURL]; } - + [picker dismissViewControllerAnimated:YES completion:NULL]; - + if(pickedImageUrl == nil && pickedVideoUrl == nil) { _result([FlutterError errorWithCode:@"file_picker_error" message:@"Temporary file could not be created" @@ -479,138 +479,150 @@ - (void)imagePickerController:(UIImagePickerController *)picker didFinishPicking _result = nil; return; } - + [self handleResult: pickedVideoUrl != nil ? pickedVideoUrl : pickedImageUrl]; } #ifdef PHPicker -(void)picker:(PHPickerViewController *)picker didFinishPicking:(NSArray *)results API_AVAILABLE(ios(14)){ - if(_result == nil) { return; } - - if(self.group != nil) { - return; - } - + Log(@"Picker:%@ didFinishPicking:%@", picker, results); - [picker dismissViewControllerAnimated:YES completion:nil]; - + if(results.count == 0) { Log(@"FilePicker canceled"); _result(nil); _result = nil; return; } - - NSMutableArray * urls = [[NSMutableArray alloc] initWithCapacity: results.count]; + + NSMutableArray * urls = [[NSMutableArray alloc] init]; + NSMutableArray * errors = [[NSMutableArray alloc] init]; self.group = dispatch_group_create(); + // Create image directory if it doesn't exist + NSString *documentsPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject]; + NSString *imagesDir = [documentsPath stringByAppendingPathComponent:@"picked_images"]; + NSFileManager *fileManager = [NSFileManager defaultManager]; + + if (![fileManager fileExistsAtPath:imagesDir]) { + NSError *dirError; + [fileManager createDirectoryAtPath:imagesDir withIntermediateDirectories:YES attributes:nil error:&dirError]; + if (dirError) { + Log(@"Failed to create image directory: %@", dirError); + } + } + if(self->_eventSink != nil) { self->_eventSink([NSNumber numberWithBool:YES]); } - - __block NSError * blockError; - - for (NSInteger index = 0; index < results.count; ++index) { - [urls addObject:[NSURL URLWithString:@""]]; + // Process images sequentially to avoid memory spikes + dispatch_queue_t processQueue = dispatch_queue_create("com.filepicker.imageprocessing", DISPATCH_QUEUE_SERIAL); + __block NSInteger completedCount = 0; + NSInteger totalCount = results.count; + + for (NSInteger index = 0; index < results.count; ++index) { dispatch_group_enter(_group); + PHPickerResult * result = [results objectAtIndex:index]; + + dispatch_async(processQueue, ^{ + @autoreleasepool { + if (![result.itemProvider hasItemConformingToTypeIdentifier:@"public.image"]) { + [errors addObject:[NSString stringWithFormat:@"Item at index %ld is not an image", (long)index]]; + dispatch_group_leave(self->_group); + return; + } - PHPickerResult * result = [results objectAtIndex: index]; + [result.itemProvider loadFileRepresentationForTypeIdentifier:@"public.image" completionHandler:^(NSURL * _Nullable url, NSError * _Nullable error) { + @autoreleasepool { + if (error != nil || url == nil) { + [errors addObject:[NSString stringWithFormat:@"Failed to load image at index %ld: %@", + (long)index, error ? error.localizedDescription : @"Unknown error"]]; + dispatch_group_leave(self->_group); + return; + } - [result.itemProvider loadFileRepresentationForTypeIdentifier:@"public.item" completionHandler:^(NSURL * _Nullable url, NSError * _Nullable error) { - - if(url == nil) { - blockError = error; - Log("Could not load the picked given file: %@", blockError); - dispatch_group_leave(self->_group); - return; - } - - long timestamp = (long)([[NSDate date] timeIntervalSince1970] * 1000); - NSString * filenameWithoutExtension = [url.lastPathComponent stringByDeletingPathExtension]; - NSString * fileExtension = url.pathExtension; - NSString * filename = [NSString stringWithFormat:@"%@-%ld.%@", filenameWithoutExtension, timestamp, fileExtension]; - NSString * extension = [filename pathExtension]; - NSFileManager * fileManager = [[NSFileManager alloc] init]; - NSURL * cachedUrl; - - // Check for live photos - if(self.allowCompression && [extension isEqualToString:@"pvt"]) { - NSArray * files = [fileManager contentsOfDirectoryAtURL:url includingPropertiesForKeys:@[] options:NSDirectoryEnumerationSkipsHiddenFiles error:nil]; - - for (NSURL * item in files) { - - if (UTTypeConformsTo(UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension, CFBridgingRetain([item pathExtension]), NULL), kUTTypeImage)) { - NSData *assetData = [NSData dataWithContentsOfURL:item]; - //Convert any type of image to jpeg - NSData *convertedImageData = UIImageJPEGRepresentation([UIImage imageWithData:assetData], 1.0); - //Get meta data from asset - NSDictionary *metaData = [ImageUtils getMetaDataFromImageData:assetData]; - //Append meta data into jpeg of live photo - NSData *data = [ImageUtils imageFromImage:convertedImageData withMetaData:metaData]; - //Save jpeg - NSString * filenameWithoutExtension = [filename stringByDeletingPathExtension]; - NSString * tmpFile = [NSTemporaryDirectory() stringByAppendingPathComponent:[filenameWithoutExtension stringByAppendingString:@".jpeg"]]; - cachedUrl = [NSURL fileURLWithPath: tmpFile]; - - if([fileManager fileExistsAtPath:tmpFile]) { - [fileManager removeItemAtPath:tmpFile error:nil]; + @try { + // Create unique filename in app_images directory + NSString *filename = [NSString stringWithFormat:@"image_%@_%ld.%@", + [[NSUUID UUID] UUIDString], + (long)[[NSDate date] timeIntervalSince1970], + url.pathExtension.length > 0 ? url.pathExtension : @"jpg"]; + + NSString *destinationPath = [imagesDir stringByAppendingPathComponent:filename]; + NSURL *destinationUrl = [NSURL fileURLWithPath:destinationPath]; + + // Load image data with options to reduce memory usage + NSError *loadError = nil; + NSData *imageData = [NSData dataWithContentsOfURL:url options:NSDataReadingMappedIfSafe error:&loadError]; + + if (loadError || !imageData) { + [errors addObject:[NSString stringWithFormat:@"Failed to load image data at index %ld: %@", + (long)index, loadError.localizedDescription ?: @"Unknown error"]]; + } else { + // Write to destination + if ([imageData writeToURL:destinationUrl options:NSDataWritingAtomic error:&loadError]) { + [urls addObject:destinationUrl]; + } else { + [errors addObject:[NSString stringWithFormat:@"Failed to save image at index %ld: %@", + (long)index, loadError.localizedDescription]]; + } + } + + // Clean up + imageData = nil; + + } @catch (NSException *exception) { + [errors addObject:[NSString stringWithFormat:@"Exception processing image at index %ld: %@", + (long)index, exception.description]]; } - if([fileManager createFileAtPath:tmpFile contents:data attributes:nil]) { - filename = tmpFile; - } else { - Log("%@ Error while caching picked Live photo", self); + // Update progress + completedCount++; + if(self->_eventSink != nil) { + dispatch_async(dispatch_get_main_queue(), ^{ + self->_eventSink(@{ + @"type": @"progress", + @"count": @(completedCount), + @"total": @(totalCount) + }); + }); } - break; + + dispatch_group_leave(self->_group); } - } - } else { - NSString * cachedFile = [NSTemporaryDirectory() stringByAppendingPathComponent:filename]; - - if([fileManager fileExistsAtPath:cachedFile]) { - [fileManager removeItemAtPath:cachedFile error:NULL]; - } - - cachedUrl = [NSURL fileURLWithPath: cachedFile]; - - NSError *copyError; - [fileManager copyItemAtURL: url - toURL: cachedUrl - error: ©Error]; - - if (copyError) { - Log("%@ Error while caching picked file: %@", self, copyError); - return; - } + }]; } - - - [urls replaceObjectAtIndex:index withObject:cachedUrl]; - dispatch_group_leave(self->_group); - }]; + }); } - + dispatch_group_notify(_group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),^{ self->_group = nil; + if(self->_eventSink != nil) { self->_eventSink([NSNumber numberWithBool:NO]); } - - if(blockError) { + + if (urls.count > 0) { + // If we have at least one successful image, return the results + if (errors.count > 0) { + // Log errors but don't fail the operation + Log(@"Some images failed to process: %@", [errors componentsJoinedByString:@", "]); + } + [self handleResult:urls]; + } else { + // Only if all images failed, return an error self->_result([FlutterError errorWithCode:@"file_picker_error" - message:@"Temporary file could not be created" - details:blockError.description]); - self->_result = nil; - return; + message:@"Failed to process any images" + details:[errors componentsJoinedByString:@"\n"]]); } - [self handleResult:urls]; + self->_result = nil; }); } @@ -665,10 +677,8 @@ - (void)presentationControllerDidDismiss:(UIPresentationController *)controller #ifdef PICKER_AUDIO - (void)mediaPickerDidCancel:(MPMediaPickerController *)controller { Log(@"FilePicker canceled"); - if (self.result != nil) { - self.result(nil); - self.result = nil; - } + _result(nil); + _result = nil; [controller dismissViewControllerAnimated:YES completion:NULL]; } #endif // PICKER_AUDIO @@ -676,10 +686,8 @@ - (void)mediaPickerDidCancel:(MPMediaPickerController *)controller { #ifdef PICKER_DOCUMENT - (void)documentPickerWasCancelled:(UIDocumentPickerViewController *)controller { Log(@"FilePicker canceled"); - if (self.result != nil) { - self.result(nil); - self.result = nil; - } + _result(nil); + _result = nil; [controller dismissViewControllerAnimated:YES completion:NULL]; } #endif // PICKER_DOCUMENT @@ -687,10 +695,8 @@ - (void)documentPickerWasCancelled:(UIDocumentPickerViewController *)controller #ifdef PICKER_MEDIA - (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker { Log(@"FilePicker canceled"); - if (self.result != nil) { - self.result(nil); - self.result = nil; - } + _result(nil); + _result = nil; [picker dismissViewControllerAnimated:YES completion:NULL]; } #endif From 66f0d0df6540019240956f786b053e124b324e05 Mon Sep 17 00:00:00 2001 From: BAHADUR ZAMAN Date: Thu, 12 Dec 2024 10:48:03 +0500 Subject: [PATCH 2/4] bumped from 8.1.5 to 8.1.6 --- pubspec.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pubspec.yaml b/pubspec.yaml index d5bf21dd..f2da922b 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -3,7 +3,7 @@ description: A package that allows you to use a native file explorer to pick sin homepage: https://github.com/miguelpruivo/plugins_flutter_file_picker repository: https://github.com/miguelpruivo/flutter_file_picker issue_tracker: https://github.com/miguelpruivo/flutter_file_picker/issues -version: 8.1.5 +version: 8.1.6 dependencies: flutter: From 53edff74b028f977841f1d13c9fdefae289e5bd3 Mon Sep 17 00:00:00 2001 From: BAHADUR ZAMAN Date: Thu, 12 Dec 2024 10:49:03 +0500 Subject: [PATCH 3/4] bumped from 8.1.5 to 8.1.6 --- CHANGELOG.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6e9922fb..9a8b7e8d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,7 @@ +## 8.1.6 +### iOS +Fix Image Picker to Handle Partial Failures Gracefully [#1554](https://github.com/miguelpruivo/flutter_file_picker/issues/1554) + ## 8.1.5 ### Android Fix [#872](https://github.com/miguelpruivo/flutter_file_picker/issues/872) From c4cab397b03417a3c0eab6a190085dd85d597f4e Mon Sep 17 00:00:00 2001 From: Miguel Ruivo Date: Thu, 26 Dec 2024 11:56:27 +0000 Subject: [PATCH 4/4] Update CHANGELOG.md --- CHANGELOG.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4ce50655..f3212d2b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,7 +1,8 @@ -## 8.1.6 +## 8.1.7 ### iOS Fix Image Picker to Handle Partial Failures Gracefully [#1554](https://github.com/miguelpruivo/flutter_file_picker/issues/1554) +## 8.1.6 ### Android Fix [#1643](https://github.com/miguelpruivo/flutter_file_picker/issues/1643)