我的相册中有一张gif图片。当我使用UIImagePickerController
选择该图像时,我需要将图像转换为NSData
进行存储。
早些时候,我用过
NSData *thumbData = UIImageJPEGRepresentation(thumbnail, 0.5);
但它不适用于gif图像。 thumbData
将为零。
如何从gif图片中获取NSData
?
我怎么知道这是一个需要特殊处理的gif图像呢?
答案 0 :(得分:9)
这里的关键是将GIF文件或URL下载直接保存到NSData中,而不是将其作为UIImage。绕过UIImage会让GIF文件保留动画。
以下是将GIF文件转换为NSData的一些代码:
NSString *filePath = [[NSBundle mainBundle] pathForResource: @"gifFileName" ofType: @"gif"];
NSData *gifData = [NSData dataWithContentsOfFile: filePath];
但说实话,你应该考虑不使用GIF。
答案 1 :(得分:2)
Gif必须不在资产中。
let path = Bundle.main.path(forResource: "loader", ofType: "gif")!
let data = try! Data(contentsOf: URL(fileURLWithPath: path))
return data
答案 2 :(得分:1)
要转换的代码.GIF文件可以在NSdata中转换 -
NSString *pathForFile = [[NSBundle mainBundle] pathForResource: @"myGif" ofType: @"gif"];
NSData *dataOfGif = [NSData dataWithContentsOfFile: pathForFile];
NSLog(@"Data: %@", dataOfGif);
答案 3 :(得分:1)
https://github.com/mattt/AnimatedGIFImageSerialization
UIImage *image = ...;
NSData *data = [AnimatedGIFImageSerialization animatedGIFDataWithImage:image
duration:1.0
loopCount:1
error:nil];
答案 4 :(得分:0)
import UIKit
import MobileCoreServices
extension UIImage {
func UIImageAnimatedGIFRepresentation(gifDuration: TimeInterval = 0.0, loopCount: Int = 0) throws -> Data {
let images = self.images ?? [self]
let frameCount = images.count
let frameDuration: TimeInterval = gifDuration <= 0.0 ? self.duration / Double(frameCount) : gifDuration / Double(frameCount)
let frameDelayCentiseconds = Int(lrint(frameDuration * 100))
let frameProperties = [
kCGImagePropertyGIFDictionary: [
kCGImagePropertyGIFDelayTime: NSNumber(value: frameDelayCentiseconds)
]
]
guard let mutableData = CFDataCreateMutable(nil, 0),
let destination = CGImageDestinationCreateWithData(mutableData, kUTTypeGIF, frameCount, nil) else {
throw NSError(domain: "AnimatedGIFSerializationErrorDomain",
code: -1,
userInfo: [NSLocalizedDescriptionKey: "Could not create destination with data."])
}
let imageProperties = [
kCGImagePropertyGIFDictionary: [kCGImagePropertyGIFLoopCount: NSNumber(value: loopCount)]
] as CFDictionary
CGImageDestinationSetProperties(destination, imageProperties)
for image in images {
if let cgimage = image.cgImage {
CGImageDestinationAddImage(destination, cgimage, frameProperties as CFDictionary)
}
}
let success = CGImageDestinationFinalize(destination)
if !success {
throw NSError(domain: "AnimatedGIFSerializationErrorDomain",
code: -2,
userInfo: [NSLocalizedDescriptionKey: "Could not finalize image destination"])
}
return mutableData as Data
}
}
UIImagePickerControllerDelegate
函数中的实现。func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) {
let info = Dictionary(uniqueKeysWithValues: info.map {key, value in (key.rawValue, value)})
if let url = info[UIImagePickerController.InfoKey.referenceURL.rawValue] as? URL, url.pathExtension.lowercased() == "gif" {
picker.dismiss(animated: false, completion: nil)
url.getGifImageDataFromAssetUrl(completion: { imageData in
// Use imageData here.
})
return
}
}
在 URL 中使用扩展函数
import UIKit
import Photos
extension URL {
func getGifImageDataFromAssetUrl(completion: @escaping(_ imageData: Data?) -> Void) {
let asset = PHAsset.fetchAssets(withALAssetURLs: [self], options: nil)
if let image = asset.firstObject {
PHImageManager.default().requestImageData(for: image, options: nil) { (imageData, _, _, _) in
completion(imageData)
}
}
}
}