21

You can do it sneakily† using the undocumented PHAsset.ALAssetURL property, but I'm looking for something documented.


† In Objective-C, this will help

@interface PHAsset (Sneaky)

@property (nonatomic, readonly) NSURL *ALAssetURL;

@end
Clay Bridges
  • 10,746
  • 10
  • 62
  • 110

5 Answers5

45

Create the assetURL by leveraging the localidentifier of the PHAsset. Example: PHAsset.localidentifier returns 91B1C271-C617-49CE-A074-E391BA7F843F/L0/001

Now take the 32 first characters to build the assetURL, like:

assets-library://asset/asset.JPG?id=91B1C271-C617-49CE-A074-E391BA7F843F&ext=JPG

You might change the extension JPG depending on the UTI of the asset (requestImageDataForAsset returns the UTI), but in my testing the extensions of the assetURL seems to be ignored anyhow.

Clay Bridges
  • 10,746
  • 10
  • 62
  • 110
holtmann
  • 5,957
  • 30
  • 42
  • Good one. Is this relationship documented? If not, where/how did you discover this? Did you just notice the GUID strings were the same? – Clay Bridges Mar 06 '15 at 14:07
  • 1
    Not documented, I saw that the GUID strings are the same and it makes sense as both the PhotoKit and AssetsLibrary work with the same CoreData objects under the hood - so this is just the UUID of the coredata object. I would prefer this method to using private API calls. – holtmann Mar 06 '15 at 14:09
  • Well, props. I'll hold a bit on the green check for a documented response, since that's what I asked for, but this has me much relieved. Big thanks. – Clay Bridges Mar 06 '15 at 14:13
  • I doubt that apple will provide any documented way to bridge from PhotoKit to AssetsLibrary (only vice versa). AssetsLibrary is legacy and they want developers to move to the new API. – holtmann Mar 06 '15 at 14:15
  • 1
    There is a one moment which prevents me to use this approach. It is an asset extension. Camera Roll can contain not only JPG images and not only MOV movies. – Pavel Osipov Mar 10 '15 at 08:21
  • 1
    Doesn't this assume that the PHAsset is a local asset ? – Ryan Heitner Jun 17 '15 at 09:09
  • @PavelOsipov Doesn't this answer already account for the UTI? Regardless, I'm thinking `PHAssetMediaTypeVideo` would map to `&ext=MOV`, or something like that. – Clay Bridges Sep 20 '15 at 02:58
  • 47
    It's ridiculous that we have to do this shit just to get a URL. – user3344977 Mar 17 '16 at 20:49
  • 2
    Make sure to remove the "/L0/001" from the localIdentifier or this will fail every time. – user3344977 Mar 17 '16 at 21:20
  • 6
    The proper way to get the GUID from the localIdentifier is to split on the first /. (Some GUIDs are longer than 32 characters.) If anybody know of a better way of getting an asset URL, please post here. This is hacky and prone to breaking with future updates. – sabalaba Oct 15 '16 at 00:40
  • 3
    DO NOT use this approach. It's hack and undocumented, Apple may change the rule of composing assets URL at anytime. Besides, you can't handle some edge situations like undownloaded resources. I gave it a try before, then some users reported they can't see some images from their album. I saw a lot of error reports in my crash reporter saids AssetsLibrary can't find image at URL: assets-library://xxx, cost me a lot of time to figure it out. A better approach is just extract the PHAsset to local file system and refactor your code to use file system URLs, or just migrate to PHAsset. – Pride Chung Jun 22 '17 at 03:57
  • I get the error "Value of type 'PHAsset?' has no member 'localidentifier'" – Chewie The Chorkie Jun 05 '18 at 18:03
9

I wanted to be able to get a URL for an asset too. However, I have realised that the localIdentifier can be persisted instead and used to recover the PHAsset.

PHAsset* asset = [PHAsset fetchAssetsWithLocalIdentifiers:@[localIdentifier] options:nil].firstObject;

Legacy asset URLs can be converted using:

PHAsset* legacyAsset = [PHAsset fetchAssetsWithALAssetUrls:@[assetUrl] options:nil].firstObject;
NSString* convertedIdentifier = legacyAsset.localIdentifier;

(before that method gets obsoleted...)

(Thanks holtmann - localIdentifier is hidden away in PHObject.)

Community
  • 1
  • 1
Matt
  • 3,554
  • 4
  • 32
  • 50
  • 2
    What you are saying _may_ be true, but it seems only obliquely related to the actual (I thought, clearly stated) question. – Clay Bridges May 20 '15 at 14:02
  • @Clay Bridges True. I guess I had a realisation that helped me solve my problem, and added an answer without re-checking the question. Sorry. – Matt May 21 '15 at 21:23
  • 2
    I find creating a separate self-answered question a better format for any such realizations you make. I've done it a lot, and despite what they tell you, haven't gone blind yet. :) You can even link to this question. – Clay Bridges May 22 '15 at 15:28
2

Here is working code tested on iOS 11 both simulator and device

PHFetchResult *result = [PHAsset fetchAssetsWithMediaType:PHAssetMediaTypeImage options:nil];
    [result enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        PHAsset *asset = (PHAsset *)obj;
        [asset requestContentEditingInputWithOptions:nil completionHandler:^(PHContentEditingInput * _Nullable contentEditingInput, NSDictionary * _Nonnull info) {
            NSLog(@"URL:%@",  contentEditingInput.fullSizeImageURL.absoluteString);
            NSString* path = [contentEditingInput.fullSizeImageURL.absoluteString substringFromIndex:7];//screw all the crap of file://
            NSFileManager *fileManager = [NSFileManager defaultManager];
            BOOL isExist = [fileManager fileExistsAtPath:path];
            if (isExist)
                NSLog(@"oh yeah");
            else {
                NSLog(@"damn");
            }
        }];
    }];
Lance Mao
  • 91
  • 1
  • 5
2

Read the bottom!

The resultHandler for PHImageManager.requestImage returns 2 objects: result and info

You can get the original filename for the PHAsset (like IMG_1043.JPG) as well as it's full path on the filesystem with:

let url = info?["PHImageFileURLKey"] as! URL

This should work right, but for some reason Apple fucked it up. So basically, you have to copy your image to a file then access that then delete it.

The PHImageFileURLKey is usable to get the original file name, but you cannot actually access that file. It probably has to do with the fact that code in the background can access the file while other apps can delete it.

Think how much time you would waste figuring that out since Apple fucked up rather that just documenting it. Maybe it's sincere, maybe it's that they had different developers who didn't understand and did not want to work. Here I am working though, and Apple has how many hundreds of billions in the bank?

Yea, sorry Apple but I do not agree when I have < 500 dollars and a net worth of less than than 1 year of 1700 subscribers paying for 1TB iCloud @ $10 a month for working for the same period of time that you have been revived from debt (2001 -> now). That was the original iPhone right? You sure are teaching them those data sizes though aren't you, scaling the iCloud monthly prices < 1% of the average US monthly income!

1

Here is a PHAsset extension written in Swift that will retrieve the URL.

extension PHAsset {

    func getURL(completionHandler : @escaping ((_ responseURL : URL?) -> Void)){
        if self.mediaType == .image {
            let options: PHContentEditingInputRequestOptions = PHContentEditingInputRequestOptions()
            options.canHandleAdjustmentData = {(adjustmeta: PHAdjustmentData) -> Bool in
                return true
            }
            self.requestContentEditingInput(with: options, completionHandler: {(contentEditingInput: PHContentEditingInput?, info: [AnyHashable : Any]) -> Void in
                completionHandler(contentEditingInput!.fullSizeImageURL as URL?)
            })
        } else if self.mediaType == .video {
            let options: PHVideoRequestOptions = PHVideoRequestOptions()
            options.version = .original
            PHImageManager.default().requestAVAsset(forVideo: self, options: options, resultHandler: {(asset: AVAsset?, audioMix: AVAudioMix?, info: [AnyHashable : Any]?) -> Void in
                if let urlAsset = asset as? AVURLAsset {
                    let localVideoUrl: URL = urlAsset.url as URL
                    completionHandler(localVideoUrl)
                } else {
                    completionHandler(nil)
                }
            })
        }
    }
}
Gene Z. Ragan
  • 2,094
  • 1
  • 26
  • 36