bellchen 71326e9589 change rootDic to rootDirectory;
change animationWithFile to animationWithFilePath
2017-06-12 10:56:39 +08:00

374 lines
14 KiB
Objective-C

//
// LOTLayerView.m
// LottieAnimator
//
// Created by Brandon Withrow on 12/14/15.
// Copyright © 2015 Brandon Withrow. All rights reserved.
//
#import "LOTLayerView.h"
#import "LOTShapeLayerView.h"
#import "LOTRectShapeLayer.h"
#import "LOTEllipseShapeLayer.h"
#import "LOTGroupLayerView.h"
#import "CAAnimationGroup+LOTAnimatableGroup.h"
#import "LOTMaskLayer.h"
#import "CGGeometry+LOTAdditions.h"
@interface LOTParentLayer : LOTAnimatableLayer
- (instancetype)initWithParentModel:(LOTLayer *)parent;
@end
@implementation LOTParentLayer {
LOTLayer *_parentModel;
CAAnimationGroup *_animation;
}
- (instancetype)initWithParentModel:(LOTLayer *)parent {
self = [super initWithLayerDuration:parent.layerDuration];
if (self) {
self.bounds = parent.layerBounds;
_parentModel = parent;
[self _setupLayerFromModel];
}
return self;
}
- (void)_setupLayerFromModel {
if (_parentModel.position) {
self.position = _parentModel.position.initialPoint;
} else {
CGPoint initial = CGPointZero;
if (_parentModel.positionX) {
initial.x = _parentModel.positionX.initialValue.floatValue;
}
if (_parentModel.positionY) {
initial.y = _parentModel.positionY.initialValue.floatValue;
}
self.position = initial;
}
self.anchorPoint = _parentModel.anchor.initialPoint;
self.transform = _parentModel.scale.initialScale;
self.sublayerTransform = CATransform3DMakeRotation(_parentModel.rotation.initialValue.floatValue, 0, 0, 1);
[self _buildAnimations];
}
- (void)_buildAnimations {
NSMutableDictionary *keypaths = [NSMutableDictionary dictionary];
if (_parentModel.position) {
[keypaths setValue:_parentModel.position forKey:@"position"];
}
if (_parentModel.anchor) {
[keypaths setValue:_parentModel.anchor forKey:@"anchorPoint"];
}
if (_parentModel.scale) {
[keypaths setValue:_parentModel.scale forKey:@"transform"];
}
if (_parentModel.rotation) {
[keypaths setValue:_parentModel.rotation forKey:@"sublayerTransform.rotation"];
}
if (_parentModel.positionX) {
[keypaths setValue:_parentModel.positionX forKey:@"position.x"];
}
if (_parentModel.positionY) {
[keypaths setValue:_parentModel.positionY forKey:@"position.y"];
}
_animation = [CAAnimationGroup LOT_animationGroupForAnimatablePropertiesWithKeyPaths:keypaths];
[self addAnimation:_animation forKey:@"LottieAnimation"];
}
@end
@implementation LOTLayerView {
NSArray<LOTGroupLayerView *> *_shapeLayers;
CALayer *_childContainerLayer;
CALayer *_rotationLayer;
CAAnimationGroup *_animation;
CAKeyframeAnimation *_inOutAnimation;
NSArray<LOTParentLayer *> *_parentLayers;
LOTMaskLayer *_maskLayer;
CALayer *_childSolid;
}
- (instancetype)initWithModel:(LOTLayer *)model inLayerGroup:(LOTLayerGroup *)layerGroup {
self = [super initWithLayerDuration:model.layerDuration];
if (self) {
_layerModel = model;
[self _setupViewFromModelWithLayerGroup:layerGroup];
}
return self;
}
- (void)_setupViewFromModelWithLayerGroup:(LOTLayerGroup *)layersGroup {
self.backgroundColor = nil;
self.bounds = _layerModel.layerBounds;
self.anchorPoint = CGPointZero;
_childContainerLayer = [CALayer new];
_childContainerLayer.bounds = _layerModel.layerBounds;
_childContainerLayer.backgroundColor = _layerModel.solidColor.CGColor;
if (_layerModel.layerType <= LOTLayerTypeSolid) {
self.bounds = _layerModel.parentCompBounds;
_childContainerLayer.backgroundColor = nil;
_childContainerLayer.masksToBounds = NO;
}
if (_layerModel.layerType == LOTLayerTypeSolid) {
[self _createChildSolid];
[self _setSolidLayerBackground];
}
if (_layerModel.layerType == LOTLayerTypeImage) {
[self _createChildSolid];
[self _setImageForAsset];
}
NSNumber *parentID = _layerModel.parentID;
CALayer *currentChild = _childContainerLayer;
NSMutableArray *parentLayers = [NSMutableArray array];
if (parentID) {
while (parentID != nil) {
LOTLayer *parentModel = [layersGroup layerModelForID:parentID];
LOTParentLayer *parentLayer = [[LOTParentLayer alloc] initWithParentModel:parentModel];
[parentLayer addSublayer:currentChild];
[parentLayers addObject:parentLayer];
currentChild = parentLayer;
parentID = parentModel.parentID;
}
}
if (parentLayers.count) {
_parentLayers = parentLayers;
}
[self addSublayer:currentChild];
_childContainerLayer.opacity = _layerModel.opacity.initialValue.floatValue;
if (_layerModel.position) {
_childContainerLayer.position = _layerModel.position.initialPoint;
} else {
CGPoint initial = CGPointZero;
if (_layerModel.positionX) {
initial.x = _layerModel.positionX.initialValue.floatValue;
}
if (_layerModel.positionY) {
initial.y = _layerModel.positionY.initialValue.floatValue;
}
_childContainerLayer.position = initial;
}
_childContainerLayer.anchorPoint = _layerModel.anchor.initialPoint;
_childContainerLayer.transform = _layerModel.scale.initialScale;
_childContainerLayer.sublayerTransform = CATransform3DMakeRotation(_layerModel.rotation.initialValue.floatValue, 0, 0, 1);
self.hidden = _layerModel.hasInAnimation;
NSArray *groupItems = _layerModel.shapes;
NSArray *reversedItems = [[groupItems reverseObjectEnumerator] allObjects];
LOTShapeTransform *currentTransform = [LOTShapeTransform transformIdentityWithCompBounds:_layerModel.layerBounds];
LOTShapeTrimPath *currentTrimPath = nil;
LOTShapeFill *currentFill = nil;
LOTShapeStroke *currentStroke = nil;
NSMutableArray *shapeLayers = [NSMutableArray array];
for (id item in reversedItems) {
if ([item isKindOfClass:[LOTShapeGroup class]]) {
LOTGroupLayerView *groupLayer = [[LOTGroupLayerView alloc] initWithShapeGroup:(LOTShapeGroup *)item
transform:currentTransform
fill:currentFill
stroke:currentStroke
trimPath:currentTrimPath
withLayerDuration:self.layerDuration];
[_childContainerLayer addSublayer:groupLayer];
[shapeLayers addObject:groupLayer];
} else if ([item isKindOfClass:[LOTShapePath class]]) {
LOTShapePath *shapePath = (LOTShapePath *)item;
LOTShapeLayerView *shapeLayer = [[LOTShapeLayerView alloc] initWithShape:shapePath
fill:currentFill
stroke:currentStroke
trim:currentTrimPath
transform:currentTransform
withLayerDuration:self.layerDuration];
[shapeLayers addObject:shapeLayer];
[_childContainerLayer addSublayer:shapeLayer];
} else if ([item isKindOfClass:[LOTShapeRectangle class]]) {
LOTShapeRectangle *shapeRect = (LOTShapeRectangle *)item;
LOTRectShapeLayer *shapeLayer = [[LOTRectShapeLayer alloc] initWithRectShape:shapeRect
fill:currentFill
stroke:currentStroke
trim:currentTrimPath
transform:currentTransform
withLayerDuration:self.layerDuration];
[shapeLayers addObject:shapeLayer];
[_childContainerLayer addSublayer:shapeLayer];
} else if ([item isKindOfClass:[LOTShapeCircle class]]) {
LOTShapeCircle *shapeCircle = (LOTShapeCircle *)item;
LOTEllipseShapeLayer *shapeLayer = [[LOTEllipseShapeLayer alloc] initWithEllipseShape:shapeCircle
fill:currentFill
stroke:currentStroke
trim:currentTrimPath
transform:currentTransform
withLayerDuration:self.layerDuration];
[shapeLayers addObject:shapeLayer];
[_childContainerLayer addSublayer:shapeLayer];
} else if ([item isKindOfClass:[LOTShapeTransform class]]) {
currentTransform = (LOTShapeTransform *)item;
} else if ([item isKindOfClass:[LOTShapeFill class]]) {
currentFill = (LOTShapeFill *)item;
} else if ([item isKindOfClass:[LOTShapeTrimPath class]]) {
currentTrimPath = (LOTShapeTrimPath *)item;
} else if ([item isKindOfClass:[LOTShapeStroke class]]) {
currentStroke = (LOTShapeStroke *)item;
}
}
_shapeLayers = shapeLayers;
if (_layerModel.masks) {
_maskLayer = [[LOTMaskLayer alloc] initWithMasks:_layerModel.masks inLayer:_layerModel];
_childContainerLayer.mask = _maskLayer;
}
NSMutableArray *childLayers = [NSMutableArray array];
[childLayers addObjectsFromArray:_parentLayers];
[childLayers addObjectsFromArray:_shapeLayers];
if (_maskLayer) {
[childLayers addObject:_maskLayer];
}
[self _buildAnimations];
}
- (void)_buildAnimations {
NSMutableDictionary *keypaths = [NSMutableDictionary dictionary];
if (_layerModel.opacity) {
[keypaths setValue:_layerModel.opacity forKey:@"opacity"];
}
if (_layerModel.position) {
[keypaths setValue:_layerModel.position forKey:@"position"];
}
if (_layerModel.anchor) {
[keypaths setValue:_layerModel.anchor forKey:@"anchorPoint"];
}
if (_layerModel.scale) {
[keypaths setValue:_layerModel.scale forKey:@"transform"];
}
if (_layerModel.rotation) {
[keypaths setValue:_layerModel.rotation forKey:@"sublayerTransform.rotation"];
}
if (_layerModel.positionX) {
[keypaths setValue:_layerModel.positionX forKey:@"position.x"];
}
if (_layerModel.positionY) {
[keypaths setValue:_layerModel.positionY forKey:@"position.y"];
}
_animation = [CAAnimationGroup LOT_animationGroupForAnimatablePropertiesWithKeyPaths:keypaths];
if (_animation) {
[_childContainerLayer addAnimation:_animation forKey:@"LottieAnimation"];
}
CAKeyframeAnimation *inOutAnimation = [CAKeyframeAnimation animationWithKeyPath:@"hidden"];
inOutAnimation.keyTimes = _layerModel.inOutKeyTimes;
inOutAnimation.values = _layerModel.inOutKeyframes;
inOutAnimation.duration = _layerModel.layerDuration;
inOutAnimation.calculationMode = kCAAnimationDiscrete;
inOutAnimation.fillMode = kCAFillModeBoth;
inOutAnimation.removedOnCompletion = NO;
_inOutAnimation = inOutAnimation;
_inOutAnimation.duration = self.layerDuration;
[self addAnimation:_inOutAnimation forKey:@"inout"];
self.duration = self.layerDuration + LOT_singleFrameTimeValue;
}
- (void)LOT_addChildLayer:(CALayer *)childLayer {
[_childContainerLayer addSublayer:childLayer];
}
- (void)setDebugModeOn:(BOOL)debugModeOn {
_debugModeOn = debugModeOn;
self.borderColor = debugModeOn ? [UIColor redColor].CGColor : nil;
self.borderWidth = debugModeOn ? 2 : 0;
self.backgroundColor = debugModeOn ? [[UIColor blueColor] colorWithAlphaComponent:0.2].CGColor : [UIColor clearColor].CGColor;
_childContainerLayer.borderColor = debugModeOn ? [UIColor yellowColor].CGColor : nil;
_childContainerLayer.borderWidth = debugModeOn ? 2 : 0;
_childContainerLayer.backgroundColor = debugModeOn ? [[UIColor orangeColor] colorWithAlphaComponent:0.2].CGColor : [UIColor clearColor].CGColor;
for (LOTGroupLayerView *group in _shapeLayers) {
group.debugModeOn = debugModeOn;
}
}
- (NSString*)description {
NSMutableString *text = [[super description] mutableCopy];
[text appendFormat:@" model: %@", _layerModel];
return text;
}
#if TARGET_OS_IPHONE || TARGET_IPHONE_SIMULATOR
- (void)_setImageForAsset {
if (_layerModel.imageAsset.imageName) {
UIImage *image;
if (_layerModel.imageAsset.rootDirectory.length > 0) {
NSString *rootDirectory = _layerModel.imageAsset.rootDirectory;
if (_layerModel.imageAsset.imageDirectory.length > 0) {
rootDirectory = [rootDirectory stringByAppendingPathComponent:_layerModel.imageAsset.imageDirectory];
}
NSString *imagePath = [rootDirectory stringByAppendingPathComponent:_layerModel.imageAsset.imageName];
image = [UIImage imageWithContentsOfFile:imagePath];
}else{
NSArray *components = [_layerModel.imageAsset.imageName componentsSeparatedByString:@"."];
image = [UIImage imageNamed:components.firstObject];
}
if (image) {
_childSolid.contents = (__bridge id _Nullable)(image.CGImage);
} else {
NSLog(@"%s: Warn: image not found: %@", __PRETTY_FUNCTION__, _layerModel.imageAsset.imageName);
}
}
}
#else
- (void)_setImageForAsset {
if (_layerModel.imageAsset.imageName) {
NSArray *components = [_layerModel.imageAsset.imageName componentsSeparatedByString:@"."];
NSImage *image = [NSImage imageNamed:components.firstObject];
if (image) {
NSWindow *window = [NSApp mainWindow];
CGFloat desiredScaleFactor = [window backingScaleFactor];
CGFloat actualScaleFactor = [image recommendedLayerContentsScale:desiredScaleFactor];
id layerContents = [image layerContentsForContentsScale:actualScaleFactor];
_childSolid.contents = layerContents;
}
}
}
#endif
- (void)_createChildSolid {
_childSolid = [CALayer new];
_childSolid.frame = _childContainerLayer.bounds;
_childSolid.masksToBounds = YES;
[_childContainerLayer addSublayer:_childSolid];
}
- (void)_setSolidLayerBackground {
_childSolid.backgroundColor = _layerModel.solidColor.CGColor;
}
@end