В моем приложении мне нужно загрузить фотографии на сервер, поэтому перед этим я хочу изменить их размер и сжать до приемлемого размера. Я пытался изменить их размер двумя способами, и первый из них:
// image is an instance of original UIImage that I want to resize
let width : Int = 640
let height : Int = 640
let bitsPerComponent = CGImageGetBitsPerComponent(image.CGImage)
let bytesPerRow = CGImageGetBytesPerRow(image.CGImage)
let colorSpace = CGImageGetColorSpace(image.CGImage)
let bitmapInfo = CGImageGetBitmapInfo(image.CGImage)
let context = CGBitmapContextCreate(nil, width, height, bitsPerComponent, bytesPerRow, colorSpace, bitmapInfo)
CGContextSetInterpolationQuality(context, kCGInterpolationHigh)
CGContextDrawImage(context, CGRect(origin: CGPointZero, size: CGSize(width: CGFloat(width), height: CGFloat(height))), image.CGImage)
image = UIImage(CGImage: CGBitmapContextCreateImage(context))
Другой способ:
image = RBResizeImage(image, targetSize: CGSizeMake(640, 640))
func RBResizeImage(image: UIImage?, targetSize: CGSize) -> UIImage? {
if let image = image {
let size = image.size
let widthRatio = targetSize.width / image.size.width
let heightRatio = targetSize.height / image.size.height
// Figure out what our orientation is, and use that to form the rectangle
var newSize: CGSize
if(widthRatio > heightRatio) {
newSize = CGSizeMake(size.width heightRatio, size.height heightRatio)
} else {
newSize = CGSizeMake(size.width widthRatio, size.height widthRatio)
}
// This is the rect that we've calculated out and this is what is actually used below
let rect = CGRectMake(0, 0, newSize.width, newSize.height)
// Actually do the resizing to the rect using the ImageContext stuff
UIGraphicsBeginImageContextWithOptions(newSize, false, 1.0)
image.drawInRect(rect)
let newImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
return newImage
} else {
return nil
}
}
После этого я использую UIImageJPEGRepresentation
для сжатия UIImage, но даже если compressionQuality
равно 1, фотография все равно размыта (в основном это видно по краям объекта, возможно, это не имеет большого значения, но фотография в три-пять раз больше, чем та же фотография из Instagram, например, но не такой резкости). Для 0.5, конечно, еще хуже, да и фото все равно больше (в КБ), чем такое же фото из Instagram.
Фотография из моего app, качество сжатия - 1, края размытые, размер - 341 КБ.
Фотография из Instagram, края резкие, и размер 136 КБ
РЕДАКТИРОВАТЬ:
Хорошо, но я сейчас немного запутался, я не уверен, что делать, чтобы сохранить соотношение сторон? Вот как я обрезаю изображение (scrollView имеет UIImageView, поэтому я могу перемещать и масштабировать изображение, а в конце я могу обрезать видимую часть scrollView, которая является sqare). Как бы то ни было, изображение сверху изначально было 2048x2048, но все равно размыто.
var scale = 1/scrollView.zoomScale
var visibleRect : CGRect = CGRect()
visibleRect.origin.x = scrollView.contentOffset.x * scale
visibleRect.origin.y = scrollView.contentOffset.y * scale
visibleRect.size.width = scrollView.bounds.size.width * scale
visibleRect.size.height = scrollView.bounds.size.height * scale
image = crop(image!, rect: visibleRect)
func crop(srcImage : UIImage, rect : CGRect) -> UIImage? {
var imageRef = CGImageCreateWithImageInRect(srcImage.CGImage, rect)
var cropped = UIImage(CGImage: imageRef)
return cropped
}