// // SVGAPlayer.m // SVGAPlayer // // Created by 崔明辉 on 16/6/17. // Copyright © 2016年 UED Center. All rights reserved. // #import "SVGAPlayer.h" #import "SVGAVideoEntity.h" #import "SVGAVideoSpriteEntity.h" #import "SVGAVideoSpriteFrameEntity.h" #import "SVGAContentLayer.h" #import "SVGABitmapLayer.h" #import "SVGAVectorLayer.h" @interface SVGAPlayer () { int _loopCount; } @property (nonatomic, strong) CALayer *drawLayer; @property (nonatomic, strong) CADisplayLink *displayLink; @property (nonatomic, assign) NSInteger currentFrame; @property (nonatomic, copy) NSDictionary *dynamicObjects; @property (nonatomic, copy) NSDictionary *dynamicTexts; @end @implementation SVGAPlayer - (instancetype)initWithFrame:(CGRect)frame { self = [super initWithFrame:frame]; if (self) { self.contentMode = UIViewContentModeTop; } return self; } - (void)willMoveToSuperview:(UIView *)newSuperview { [super willMoveToSuperview:newSuperview]; if (newSuperview == nil) { [self stopAnimation:YES]; } } - (void)startAnimation { [self stopAnimation:NO]; _loopCount = 0; self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(next)]; self.displayLink.frameInterval = 60 / self.videoItem.FPS; [self.displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes]; } - (void)pauseAnimation { [self stopAnimation:NO]; } - (void)stopAnimation { [self stopAnimation:self.clearsAfterStop]; } - (void)stopAnimation:(BOOL)clear { if (![self.displayLink isPaused]) { [self.displayLink setPaused:YES]; [self.displayLink removeFromRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes]; } if (clear) { [self clear]; } self.displayLink = nil; } - (void)clear { [self.drawLayer removeFromSuperlayer]; } - (void)stepToFrame:(NSInteger)frame andPlay:(BOOL)andPlay { if (frame >= self.videoItem.frames || frame < 0) { return; } [self pauseAnimation]; self.currentFrame = frame; [self update]; if (andPlay) { self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(next)]; self.displayLink.frameInterval = 60 / self.videoItem.FPS; [self.displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes]; } } - (void)stepToPercentage:(CGFloat)percentage andPlay:(BOOL)andPlay { NSInteger frame = (NSInteger)(self.videoItem.frames * percentage); if (frame >= self.videoItem.frames && frame > 0) { frame = self.videoItem.frames - 1; } [self stepToFrame:frame andPlay:andPlay]; } - (void)draw { self.drawLayer = [[CALayer alloc] init]; self.drawLayer.frame = CGRectMake(0, 0, self.videoItem.videoSize.width, self.videoItem.videoSize.height); self.drawLayer.masksToBounds = true; [self.videoItem.sprites enumerateObjectsUsingBlock:^(SVGAVideoSpriteEntity * _Nonnull sprite, NSUInteger idx, BOOL * _Nonnull stop) { UIImage *bitmap; if (sprite.imageKey != nil) { if (self.dynamicObjects[sprite.imageKey] != nil) { bitmap = self.dynamicObjects[sprite.imageKey]; } else { bitmap = self.videoItem.images[sprite.imageKey]; } } SVGAContentLayer *contentLayer = [sprite requestLayerWithBitmap:bitmap]; contentLayer.imageKey = sprite.imageKey; [self.drawLayer addSublayer:contentLayer]; if (sprite.imageKey != nil) { if (self.dynamicTexts[sprite.imageKey] != nil) { NSAttributedString *text = self.dynamicTexts[sprite.imageKey]; CGSize size = [text boundingRectWithSize:CGSizeMake(CGFLOAT_MAX, CGFLOAT_MAX) options:NSStringDrawingUsesLineFragmentOrigin context:NULL].size; CATextLayer *textLayer = [CATextLayer layer]; textLayer.contentsScale = [[UIScreen mainScreen] scale]; [textLayer setString:self.dynamicTexts[sprite.imageKey]]; textLayer.frame = CGRectMake(0, 0, size.width, size.height); [contentLayer addSublayer:textLayer]; contentLayer.textLayer = textLayer; } } }]; [self.layer addSublayer:self.drawLayer]; self.currentFrame = 0; [self update]; [self resize]; } - (void)resize { if (self.contentMode == UIViewContentModeScaleAspectFit) { CGFloat videoRatio = self.videoItem.videoSize.width / self.videoItem.videoSize.height; CGFloat layerRatio = self.bounds.size.width / self.bounds.size.height; if (videoRatio > layerRatio) { CGFloat ratio = self.bounds.size.width / self.videoItem.videoSize.width; CGPoint offset = CGPointMake( (1.0 - ratio) / 2.0 * self.videoItem.videoSize.width, (1.0 - ratio) / 2.0 * self.videoItem.videoSize.height - (self.bounds.size.height - self.videoItem.videoSize.height * ratio) / 2.0 ); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(ratio, 0, 0, ratio, -offset.x, -offset.y)); } else { CGFloat ratio = self.bounds.size.height / self.videoItem.videoSize.height; CGPoint offset = CGPointMake( (1.0 - ratio) / 2.0 * self.videoItem.videoSize.width - (self.bounds.size.width - self.videoItem.videoSize.width * ratio) / 2.0, (1.0 - ratio) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(ratio, 0, 0, ratio, -offset.x, -offset.y)); } } else if (self.contentMode == UIViewContentModeScaleAspectFill) { CGFloat videoRatio = self.videoItem.videoSize.width / self.videoItem.videoSize.height; CGFloat layerRatio = self.bounds.size.width / self.bounds.size.height; if (videoRatio < layerRatio) { CGFloat ratio = self.bounds.size.width / self.videoItem.videoSize.width; CGPoint offset = CGPointMake( (1.0 - ratio) / 2.0 * self.videoItem.videoSize.width, (1.0 - ratio) / 2.0 * self.videoItem.videoSize.height - (self.bounds.size.height - self.videoItem.videoSize.height * ratio) / 2.0 ); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(ratio, 0, 0, ratio, -offset.x, -offset.y)); } else { CGFloat ratio = self.bounds.size.height / self.videoItem.videoSize.height; CGPoint offset = CGPointMake( (1.0 - ratio) / 2.0 * self.videoItem.videoSize.width - (self.bounds.size.width - self.videoItem.videoSize.width * ratio) / 2.0, (1.0 - ratio) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(ratio, 0, 0, ratio, -offset.x, -offset.y)); } } else if (self.contentMode == UIViewContentModeTop) { CGFloat scaleX = self.frame.size.width / self.videoItem.videoSize.width; CGPoint offset = CGPointMake((1.0 - scaleX) / 2.0 * self.videoItem.videoSize.width, (1 - scaleX) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(scaleX, 0, 0, scaleX, -offset.x, -offset.y)); } else if (self.contentMode == UIViewContentModeBottom) { CGFloat scaleX = self.frame.size.width / self.videoItem.videoSize.width; CGPoint offset = CGPointMake( (1.0 - scaleX) / 2.0 * self.videoItem.videoSize.width, (1.0 - scaleX) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(scaleX, 0, 0, scaleX, -offset.x, -offset.y + self.frame.size.height - self.videoItem.videoSize.height * scaleX)); } else if (self.contentMode == UIViewContentModeLeft) { CGFloat scaleY = self.frame.size.height / self.videoItem.videoSize.height; CGPoint offset = CGPointMake((1.0 - scaleY) / 2.0 * self.videoItem.videoSize.width, (1 - scaleY) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(scaleY, 0, 0, scaleY, -offset.x, -offset.y)); } else if (self.contentMode == UIViewContentModeRight) { CGFloat scaleY = self.frame.size.height / self.videoItem.videoSize.height; CGPoint offset = CGPointMake( (1.0 - scaleY) / 2.0 * self.videoItem.videoSize.width, (1.0 - scaleY) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(scaleY, 0, 0, scaleY, -offset.x + self.frame.size.width - self.videoItem.videoSize.width * scaleY, -offset.y)); } else { CGFloat scaleX = self.frame.size.width / self.videoItem.videoSize.width; CGFloat scaleY = self.frame.size.height / self.videoItem.videoSize.height; CGPoint offset = CGPointMake((1.0 - scaleX) / 2.0 * self.videoItem.videoSize.width, (1 - scaleY) / 2.0 * self.videoItem.videoSize.height); self.drawLayer.transform = CATransform3DMakeAffineTransform(CGAffineTransformMake(scaleX, 0, 0, scaleY, -offset.x, -offset.y)); } } - (void)layoutSubviews { [super layoutSubviews]; [self resize]; } - (void)update { [CATransaction setDisableActions:YES]; for (SVGAContentLayer *layer in self.drawLayer.sublayers) { if ([layer isKindOfClass:[SVGAContentLayer class]]) { [layer stepToFrame:self.currentFrame]; } } [CATransaction setDisableActions:NO]; } - (void)next { self.currentFrame++; if (self.currentFrame >= self.videoItem.frames) { self.currentFrame = 0; _loopCount++; if (self.loops > 0 && _loopCount >= self.loops) { [self stopAnimation]; if (!self.clearsAfterStop && [self.fillMode isEqualToString:@"Backward"]) { [self stepToFrame:0 andPlay:NO]; } id delegate = self.delegate; if (delegate != nil && [delegate respondsToSelector:@selector(svgaPlayerDidFinishedAnimation:)]) { [delegate svgaPlayerDidFinishedAnimation:self]; } return; } } [self update]; id delegate = self.delegate; if (delegate != nil && [delegate respondsToSelector:@selector(svgaPlayerDidAnimatedToFrame:)]) { [delegate svgaPlayerDidAnimatedToFrame:self.currentFrame]; } if (delegate != nil && [delegate respondsToSelector:@selector(svgaPlayerDidAnimatedToPercentage:)] && self.videoItem.frames > 0) { [delegate svgaPlayerDidAnimatedToPercentage:(CGFloat)(self.currentFrame + 1) / (CGFloat)self.videoItem.frames]; } } - (void)setVideoItem:(SVGAVideoEntity *)videoItem { _videoItem = videoItem; [[NSOperationQueue mainQueue] addOperationWithBlock:^{ [self clear]; [self draw]; }]; } #pragma mark - Dynamic Object - (void)setImage:(UIImage *)image forKey:(NSString *)aKey { if (image == nil) { return; } NSMutableDictionary *mutableDynamicObjects = [self.dynamicObjects mutableCopy]; [mutableDynamicObjects setObject:image forKey:aKey]; self.dynamicObjects = mutableDynamicObjects; if (self.drawLayer.sublayers.count > 0) { for (SVGAContentLayer *layer in self.drawLayer.sublayers) { if ([layer isKindOfClass:[SVGAContentLayer class]] && [layer.imageKey isEqualToString:aKey]) { layer.bitmapLayer.contents = (__bridge id _Nullable)([image CGImage]); } } } } - (void)setImageWithURL:(NSURL *)URL forKey:(NSString *)aKey { [[[NSURLSession sharedSession] dataTaskWithURL:URL completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) { if (error == nil && data != nil) { UIImage *image = [UIImage imageWithData:data]; if (image != nil) { [[NSOperationQueue mainQueue] addOperationWithBlock:^{ [self setImage:image forKey:aKey]; }]; } } }] resume]; } - (void)setImage:(UIImage *)image forKey:(NSString *)aKey referenceLayer:(CALayer *)referenceLayer { [self setImage:image forKey:aKey]; } - (void)setAttributedText:(NSAttributedString *)attributedText forKey:(NSString *)aKey { if (attributedText == nil) { return; } NSMutableDictionary *mutableDynamicTexts = [self.dynamicTexts mutableCopy]; [mutableDynamicTexts setObject:attributedText forKey:aKey]; self.dynamicTexts = mutableDynamicTexts; if (self.drawLayer.sublayers.count > 0) { CGSize size = [attributedText boundingRectWithSize:CGSizeMake(CGFLOAT_MAX, CGFLOAT_MAX) options:NSStringDrawingUsesLineFragmentOrigin context:NULL].size; CATextLayer *textLayer; for (SVGAContentLayer *layer in self.drawLayer.sublayers) { if ([layer isKindOfClass:[SVGAContentLayer class]] && [layer.imageKey isEqualToString:aKey]) { textLayer = layer.textLayer; if (textLayer == nil) { textLayer = [CATextLayer layer]; [layer addSublayer:textLayer]; } } } if (textLayer != nil) { textLayer.contentsScale = [[UIScreen mainScreen] scale]; [textLayer setString:attributedText]; textLayer.frame = CGRectMake(0, 0, size.width, size.height); } } } - (void)clearDynamicObjects { self.dynamicObjects = nil; } - (NSDictionary *)dynamicObjects { if (_dynamicObjects == nil) { _dynamicObjects = @{}; } return _dynamicObjects; } - (NSDictionary *)dynamicTexts { if (_dynamicTexts == nil) { _dynamicTexts = @{}; } return _dynamicTexts; } @end