cdts/xdts-ios 3/TreeHole/CYHResetCode/CYH/QMUIKit/UIKitExtensions/UIImageView+QMUI.m

226 lines
9.9 KiB
Mathematica
Raw Permalink Normal View History

2023-07-27 09:20:00 +08:00
/**
* Tencent is pleased to support the open source community by making QMUI_iOS available.
* Copyright (C) 2016-2021 THL A29 Limited, a Tencent company. All rights reserved.
* Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
* http://opensource.org/licenses/MIT
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
*/
//
// UIImageView+QMUI.m
// qmui
//
// Created by QMUI Team on 16/8/9.
//
#import "UIImageView+QMUI.h"
#import "QMUICore.h"
#import "CALayer+QMUI.h"
#import "UIView+QMUI.h"
@interface UIImageView ()
@property(nonatomic, strong) CALayer *qimgv_animatedImageLayer;
@property(nonatomic, strong) CADisplayLink *qimgv_displayLink;
@property(nonatomic, strong) UIImage *qimgv_animatedImage;
@property(nonatomic, assign) NSInteger qimgv_currentAnimatedImageIndex;
@end
@implementation UIImageView (QMUI)
QMUISynthesizeIdStrongProperty(qimgv_animatedImageLayer, setQimgv_animatedImageLayer)
QMUISynthesizeIdStrongProperty(qimgv_displayLink, setQimgv_displayLink)
QMUISynthesizeIdStrongProperty(qimgv_animatedImage, setQimgv_animatedImage)
QMUISynthesizeNSIntegerProperty(qimgv_currentAnimatedImageIndex, setQimgv_currentAnimatedImageIndex)
- (void)qimgv_swizzleMethods {
[QMUIHelper executeBlock:^{
OverrideImplementation([UIImageView class], @selector(setImage:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UIImageView *selfObject, UIImage *image) {
// call super
void (^callSuperBlock)(UIImage *) = ^void(UIImage *aImage) {
void (*originSelectorIMP)(id, SEL, UIImage *);
originSelectorIMP = (void (*)(id, SEL, UIImage *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, aImage);
};
if (selfObject.qmui_smoothAnimation && image.images) {
if (image != selfObject.qimgv_animatedImage) {
callSuperBlock(nil);
selfObject.qimgv_animatedImage = image;
[selfObject qimgv_requestToStartAnimation];
}
} else {
selfObject.qimgv_animatedImage = nil;
[selfObject qimgv_stopAnimating];
callSuperBlock(image);
}
};
});
OverrideImplementation([UIImageView class], @selector(image), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^UIImage *(UIImageView *selfObject) {
if (selfObject.qimgv_animatedImage) {
return selfObject.qimgv_animatedImage;
}
// call super
UIImage *(*originSelectorIMP)(id, SEL);
originSelectorIMP = (UIImage *(*)(id, SEL))originalIMPProvider();
UIImage *result = originSelectorIMP(selfObject, originCMD);
return result;
};
});
ExtendImplementationOfVoidMethodWithoutArguments([UIImageView class], @selector(layoutSubviews), ^(UIImageView *selfObject) {
if (selfObject.qimgv_animatedImageLayer) {
selfObject.qimgv_animatedImageLayer.frame = selfObject.bounds;
}
});
ExtendImplementationOfVoidMethodWithoutArguments([UIImageView class], @selector(didMoveToWindow), ^(UIImageView *selfObject) {
[selfObject qimgv_updateAnimationStateAutomatically];
});
ExtendImplementationOfVoidMethodWithSingleArgument([UIImageView class], @selector(setHidden:), BOOL, ^(UIImageView *selfObject, BOOL hidden) {
[selfObject qimgv_updateAnimationStateAutomatically];
});
ExtendImplementationOfVoidMethodWithSingleArgument([UIImageView class], @selector(setAlpha:), CGFloat, ^(UIImageView *selfObject, CGFloat alpha) {
[selfObject qimgv_updateAnimationStateAutomatically];
});
ExtendImplementationOfVoidMethodWithSingleArgument([UIImageView class], @selector(setFrame:), CGRect, ^(UIImageView *selfObject, CGRect frame) {
[selfObject qimgv_updateAnimationStateAutomatically];
});
OverrideImplementation([UIImageView class], @selector(sizeThatFits:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^CGSize(UIImageView *selfObject, CGSize size) {
if (selfObject.qimgv_animatedImage) {
return selfObject.qimgv_animatedImage.size;
}
// call super
CGSize (*originSelectorIMP)(id, SEL, CGSize);
originSelectorIMP = (CGSize (*)(id, SEL, CGSize))originalIMPProvider();
CGSize result = originSelectorIMP(selfObject, originCMD, size);
return result;
};
});
ExtendImplementationOfVoidMethodWithSingleArgument([UIImageView class], @selector(setContentMode:), UIViewContentMode, ^(UIImageView *selfObject, UIViewContentMode firstArgv) {
if (selfObject.qimgv_animatedImageLayer) {
selfObject.qimgv_animatedImageLayer.contentsGravity = [QMUIHelper layerContentsGravityWithContentMode:firstArgv];
}
});
} oncePerIdentifier:@"UIImageView (QMUI) smoothAnimation"];
}
- (BOOL)qimgv_requestToStartAnimation {
if (![self qimgv_canStartAnimation]) return NO;
if (!self.qimgv_animatedImageLayer) {
self.qimgv_animatedImageLayer = [CALayer layer];
self.qimgv_animatedImageLayer.contentsGravity = [QMUIHelper layerContentsGravityWithContentMode:self.contentMode];
[self.layer addSublayer:self.qimgv_animatedImageLayer];
}
if (!self.qimgv_displayLink) {
self.qimgv_displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)];
[self.qimgv_displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
NSInteger preferredFramesPerSecond = self.qimgv_animatedImage.images.count / self.qimgv_animatedImage.duration;
self.qimgv_displayLink.preferredFramesPerSecond = preferredFramesPerSecond;
self.qimgv_currentAnimatedImageIndex = -1;
self.qimgv_animatedImageLayer.contents = (__bridge id)self.qimgv_animatedImage.images.firstObject.CGImage;// pause displayLayer:
}
self.qimgv_displayLink.paused = self.qmui_pause;
return YES;
}
- (void)qimgv_stopAnimating {
if (self.qimgv_displayLink) {
[self.qimgv_displayLink invalidate];
self.qimgv_displayLink = nil;
}
if (self.qimgv_animatedImageLayer) {
[self.qimgv_animatedImageLayer removeFromSuperlayer];
self.qimgv_animatedImageLayer = nil;
}
}
- (void)qimgv_updateAnimationStateAutomatically {
if (self.qimgv_animatedImage) {
if (![self qimgv_requestToStartAnimation]) {
[self qimgv_stopAnimating];
}
}
}
- (BOOL)qimgv_canStartAnimation {
return self.qmui_visible && !CGRectIsEmpty(self.frame);
}
- (void)handleDisplayLink:(CADisplayLink *)displayLink {
self.qimgv_currentAnimatedImageIndex = self.qimgv_currentAnimatedImageIndex < self.qimgv_animatedImage.images.count - 1 ? (self.qimgv_currentAnimatedImageIndex + 1) : 0;
self.qimgv_animatedImageLayer.contents = (__bridge id)self.qimgv_animatedImage.images[self.qimgv_currentAnimatedImageIndex].CGImage;
}
static char kAssociatedObjectKey_smoothAnimation;
- (void)setQmui_smoothAnimation:(BOOL)qmui_smoothAnimation {
objc_setAssociatedObject(self, &kAssociatedObjectKey_smoothAnimation, @(qmui_smoothAnimation), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
if (qmui_smoothAnimation) {
[self qimgv_swizzleMethods];
}
if (qmui_smoothAnimation && self.image.images && self.image != self.qimgv_animatedImage) {
self.image = self.image;//
} else if (!qmui_smoothAnimation && self.qimgv_animatedImage) {
self.image = self.image;// setImage
}
}
- (BOOL)qmui_smoothAnimation {
return [((NSNumber *)objc_getAssociatedObject(self, &kAssociatedObjectKey_smoothAnimation)) boolValue];
}
static char kAssociatedObjectKey_pause;
- (void)setQmui_pause:(BOOL)qmui_pause {
objc_setAssociatedObject(self, &kAssociatedObjectKey_pause, @(qmui_pause), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
if (self.animationImages || self.image.images) {
self.qimgv_animatedImageLayer.qmui_pause = qmui_pause;
}
if (self.qimgv_displayLink) {
self.qimgv_displayLink.paused = qmui_pause;
}
}
- (BOOL)qmui_pause {
return [((NSNumber *)objc_getAssociatedObject(self, &kAssociatedObjectKey_pause)) boolValue];
}
- (void)qmui_sizeToFitKeepingImageAspectRatioInSize:(CGSize)limitSize {
if (!self.image) {
return;
}
CGSize currentSize = self.frame.size;
if (currentSize.width <= 0) {
currentSize.width = self.image.size.width;
}
if (currentSize.height <= 0) {
currentSize.height = self.image.size.height;
}
CGFloat horizontalRatio = limitSize.width / currentSize.width;
CGFloat verticalRatio = limitSize.height / currentSize.height;
CGFloat ratio = fmin(horizontalRatio, verticalRatio);
CGRect frame = self.frame;
frame.size.width = flat(currentSize.width * ratio);
frame.size.height = flat(currentSize.height * ratio);
self.frame = frame;
}
@end