| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 |
- //
- // 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
|