前言

  上一篇已经介绍了核心动画在UI渲染中的位置和基本概念,但是没有具体介绍CAAnimation子类的用法,本文将介绍CABasicAnimation及其子类CASpringAnimation的用法和一些注意事项。

一、CABasicAnimation

1.什么是CABasicAnimation

  CABasicAnimation是核心动画类簇中的一个类,其父类是CAPropertyAnimation,其子类是CASpringAnimation,它的祖父是CAAnimation。它主要用于制作比较单一的动画,例如,平移、缩放、旋转、颜色渐变、边框的值的变化等,也就是将layer的某个属性值从一个值到另一个值的变化。类似x -> y这种变化,然而对于x -> y -> z甚至更多的变化是不行的。

2.常用属性

  @property(nullable, strong) id fromValue;

  @property(nullable, strong) id toValue;

  @property(nullable, strong) id byValue;

  很明显,fromvalue表示初始状态,tovalue表示最终状态,byvalue是在fromvalue的基础上发生的变化,这个可以慢慢测试,主要还是from和to。

3.实例化方法

  + (instancetype)animationWithKeyPath:(nullable NSString *)path;

  下面是对keypath比较全面的总结,每个keypath对应一种属性值的变化,其中涉及到颜色变化的都必须使用CGColor,因为对应是对layer的属性。

  1. #ifndef AnimationKeyPathName_h
  2. #define AnimationKeyPathName_h
  3. #import <Foundation/Foundation.h>
  4. /* CATransform3D Key Paths */
  5. /* 旋转x,y,z分别是绕x,y,z轴旋转 */
  6. static NSString *kCARotation = @"transform.rotation";
  7. static NSString *kCARotationX = @"transform.rotation.x";
  8. static NSString *kCARotationY = @"transform.rotation.y";
  9. static NSString *kCARotationZ = @"transform.rotation.z";
  10.  
  11. /* 缩放x,y,z分别是对x,y,z方向进行缩放 */
  12. static NSString *kCAScale = @"transform.scale";
  13. static NSString *kCAScaleX = @"transform.scale.x";
  14. static NSString *kCAScaleY = @"transform.scale.y";
  15. static NSString *kCAScaleZ = @"transform.scale.z";
  16.  
  17. /* 平移x,y,z同上 */
  18. static NSString *kCATranslation = @"transform.translation";
  19. static NSString *kCATranslationX = @"transform.translation.x";
  20. static NSString *kCATranslationY = @"transform.translation.y";
  21. static NSString *kCATranslationZ = @"transform.translation.z";
  22.  
  23. /* 平面 */
  24. /* CGPoint中心点改变位置,针对平面 */
  25. static NSString *kCAPosition = @"position";
  26. static NSString *kCAPositionX = @"position.x";
  27. static NSString *kCAPositionY = @"position.y";
  28.  
  29. /* CGRect */
  30. static NSString *kCABoundsSize = @"bounds.size";
  31. static NSString *kCABoundsSizeW = @"bounds.size.width";
  32. static NSString *kCABoundsSizeH = @"bounds.size.height";
  33. static NSString *kCABoundsOriginX = @"bounds.origin.x";
  34. static NSString *kCABoundsOriginY = @"bounds.origin.y";
  35.  
  36. /* 透明度 */
  37. static NSString *kCAOpacity = @"opacity";
  38. /* 背景色 */
  39. static NSString *kCABackgroundColor = @"backgroundColor";
  40. /* 圆角 */
  41. static NSString *kCACornerRadius = @"cornerRadius";
  42. /* 边框 */
  43. static NSString *kCABorderWidth = @"borderWidth";
  44. /* 阴影颜色 */
  45. static NSString *kCAShadowColor = @"shadowColor";
  46. /* 偏移量CGSize */
  47. static NSString *kCAShadowOffset = @"shadowOffset";
  48. /* 阴影透明度 */
  49. static NSString *kCAShadowOpacity = @"shadowOpacity";
  50. /* 阴影圆角 */
  51. static NSString *kCAShadowRadius = @"shadowRadius";
  52. #endif /* AnimationKeyPathName_h */

4.简单用法

  下面的类是用来测试不同keypath对应的动画,有些省略了,因为原理都是一样的。

  1.  

  #define SCREEN_WIDTH [UIScreen mainScreen].bounds.size.width

  #define SCREEN_HEIGHT [UIScreen mainScreen].bounds.size.height

  1. #import "BaseAnimationView.h"
  2.  
  3. @interface BaseAnimationView()<UITableViewDelegate,UITableViewDataSource>
  4. @property (nonatomic, strong) UIView *squareView;
  5. @property (nonatomic, strong) UILabel *squareLabel;
  6. @property (nonatomic, strong) UITableView *tableView;
  7. @property (nonatomic, strong) NSMutableArray *dataSource;
  8. @end
  9.  
  10. @implementation BaseAnimationView
  11.  
  12. - (instancetype)init{
  13. if (self = [super initWithFrame:CGRectMake(, , SCREEN_WIDTH, SCREEN_HEIGHT)]) {
  14. [self initData];
  15. [self setupUI];
  16. }
  17. return self;
  18. }
  19. - (void)setupUI{
  20. self.squareView = [[UIView alloc] initWithFrame:CGRectMake(, , , )];
  21. self.squareView.backgroundColor = [UIColor cyanColor];
  22. self.squareView.layer.borderColor = [UIColor redColor].CGColor;
  23. self.squareView.center = CGPointMake(SCREEN_WIDTH/2.0, );
  24. self.squareView.layer.shadowOpacity = 0.6;
  25. self.squareView.layer.shadowOffset = CGSizeMake(, );
  26. self.squareView.layer.shadowRadius = ;
  27. self.squareView.layer.shadowColor = [UIColor redColor].CGColor;
  28. [self addSubview:self.squareView];
  29.  
  30. self.squareLabel = [[UILabel alloc] initWithFrame:self.squareView.bounds];
  31. self.squareLabel.text = @"label";
  32. self.squareLabel.textAlignment = NSTextAlignmentCenter;
  33. self.squareLabel.textColor = [UIColor blackColor];
  34. self.squareLabel.font = [UIFont systemFontOfSize:];
  35. [self.squareView addSubview:self.squareLabel];
  36.  
  37. self.tableView = [[UITableView alloc] initWithFrame:CGRectMake(, , SCREEN_WIDTH, SCREEN_HEIGHT-) style:UITableViewStylePlain];
  38. self.tableView.delegate = self;
  39. self.tableView.dataSource = self;
  40. [self addSubview:self.tableView];
  41. }
  42.  
  43. - (CABasicAnimation *)getAnimationKeyPath:(NSString *)keyPath fromValue:(id)fromValue toValue:(id)toValue{
  44. CABasicAnimation *basicAnimation = [CABasicAnimation animationWithKeyPath:keyPath];
  45. basicAnimation.fromValue = fromValue;
  46. /*byvalue是在fromvalue的值的基础上增加量*/
  47. //basicAnimation.byValue = @1;
  48. basicAnimation.toValue = toValue;
  49. basicAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionLinear];;
  50. basicAnimation.duration = ;
  51. basicAnimation.repeatCount = ;
  52. /* animation remove from view after animation finish */
  53. basicAnimation.removedOnCompletion = YES;
  54. return basicAnimation;
  55. }
  56. - (void)initData{
  57. /*
  58. kCAScaleZ 缩放z 没有意义,因为是平面图形
  59. kCAPositionX设置y没有意义,可以随意设置,同理kCAPositionY设置x没有意义
  60. kCABackgroundColor,颜色变化必须要用CGColor
  61. 用到shadow的几个属性变化的时候,需要先设置shadow
  62. */
  63. NSValue *startPoint = [NSValue valueWithCGPoint:self.squareView.center];
  64. NSValue *endPoint = [NSValue valueWithCGPoint:CGPointMake(, )];
  65. NSValue *shadowStartPoint = [NSValue valueWithCGPoint:CGPointMake(, )];
  66. NSValue *shadowEndPoint = [NSValue valueWithCGPoint:CGPointMake(, )];
  67. id startColor = (id)([UIColor cyanColor].CGColor);
  68. id endColor = (id)([UIColor redColor].CGColor);
  69. id shadowStartColor = (id)[UIColor clearColor].CGColor;
  70. id shadowEndColor = (id)[UIColor redColor].CGColor;
  71. self.dataSource = [NSMutableArray array];
  72. NSArray *keypaths = @[kCARotation,kCARotationX,kCARotationY,kCARotationZ,
  73. kCAScale,kCAScaleX,kCAScaleZ,kCAPositionX,
  74. kCABoundsSizeW,kCAOpacity,kCABackgroundColor,kCACornerRadius,
  75. kCABorderWidth,kCAShadowColor,kCAShadowRadius,kCAShadowOffset];
  76.  
  77. NSArray *fromValues = @[@,@,@,@,
  78. @,@,@,startPoint,
  79. @,@,startColor,@,
  80. @,shadowStartColor,@,shadowStartPoint];
  81.  
  82. NSArray *toValues = @[@(M_PI),@(M_PI),@(M_PI),@(M_PI),
  83. @,@,@,endPoint,
  84. @,@,endColor,@,
  85. @,shadowEndColor,@,shadowEndPoint];
  86. for (int i=; i<keypaths.count; i++) {
  87. AnimationModel *model = [[AnimationModel alloc] init];
  88. model.keyPaths = keypaths[i];
  89. model.fromValue = fromValues[i];
  90. model.toValue = toValues[i];
  91. [self.dataSource addObject:model];
  92. }
  93. }
  94. - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
  95. return ;
  96. }
  97. - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
  98. return [self.dataSource count];
  99. }
  100. - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
  101. AnimationModel *model = [self.dataSource objectAtIndex:indexPath.row];
  102. UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"CellID"];
  103. if (cell==nil) {
  104. cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"CellID"];
  105. }
  106. cell.textLabel.text = model.keyPaths;
  107. cell.selectionStyle = ;
  108. return cell;
  109. }
  110. - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
  111. AnimationModel *model = [self.dataSource objectAtIndex:indexPath.row];
  112. CABasicAnimation *animation = [self getAnimationKeyPath:model.keyPaths fromValue:model.fromValue toValue:model.toValue];
  113. [self.squareView.layer addAnimation:animation forKey:nil];
  114. }
  115. @end

  之所以在执行动画的view上加入一个label,是为了测试当layer放大时其子类的变化。layer进行缩放时,它的大小并没有发生改变,我们看到它变大或变小是因为它离我们的距离在发生改变,所以其子视图会随着一起变化。然而,改变其frame,它的大小和坐标是确实发生了改变,其子视图正常情况下是不会发生变化的。CABasicAnimation用法非常简单,下面介绍其子类CASpringAnimation。

二、CASpringAnimation

1.什么是CASpringAnimation

  CASpringAnimation是在CABasicAnimation的基础上衍生的另一个动画类,它比CABasicAnimation多了动画的弹性,是动画不再是从一个状态变成另一个状态时显得生硬。CASpringAnimation是iOS9.0之后新加的。

2.新增属性

  @property CGFloat mass;       //质量(影响弹簧的惯性,质量越大,弹簧惯性越大,运动的幅度越大)

  @property CGFloat stiffness;  //弹性系数(弹性系数越大,弹簧的运动越快)

  @property CGFloat damping;    //阻尼系数(阻尼系数越大,弹簧的停止越快)

  @property CGFloat initialVelocity;  //初始速率(弹簧动画的初始速度大小,弹簧运动的初始方向与初始速率的正负一致,若初始速率为0,表示忽略该属性)

  @property CGFloat settlingDuration; //结算时间(根据动画参数估算弹簧开始运动到停止的时间,动画设置的时间最好根据此时间来设置)

  这三个属性可以设置动画在执行到最终状态后的弹性效果,具体值需要调试,下面给出一个具体的类来实现这一功能。

  1. #import "SpringAnimationView.h"
  2. #import "UIView+HPAdditions.h"
  3.  
  4. @interface SpringAnimationView()<UITableViewDelegate,UITableViewDataSource>
  5. @property (nonatomic, strong) UIView *squareView;
  6. @property (nonatomic, strong) UITableView *tableView;
  7. @property (nonatomic, strong) NSMutableArray *dataSource;
  8. @property (nonatomic, strong) UISlider *massSlider;
  9. @property (nonatomic, strong) UISlider *stiffnessSlider;
  10. @property (nonatomic, strong) UISlider *dampingSlider;
  11. @property (nonatomic, strong) UILabel *massLabel;
  12. @property (nonatomic, strong) UILabel *stiffnessLabel;
  13. @property (nonatomic, strong) UILabel *dampingLabel;
  14. @end
  15.  
  16. @implementation SpringAnimationView
  17.  
  18. - (instancetype)init{
  19. if (self = [super initWithFrame:CGRectMake(, , SCREEN_WIDTH, SCREEN_HEIGHT)]) {
  20. [self initData];
  21. [self setupUI];
  22. }
  23. return self;
  24. }
  25. - (void)setupUI{
  26. self.squareView = [[UIView alloc] initWithFrame:CGRectMake(, , , )];
  27. self.squareView.backgroundColor = [UIColor cyanColor];
  28. self.squareView.layer.borderColor = [UIColor redColor].CGColor;
  29. self.squareView.center = CGPointMake(SCREEN_WIDTH/2.0, );
  30. self.squareView.layer.shadowOpacity = 0.6;
  31. self.squareView.layer.shadowOffset = CGSizeMake(, );
  32. self.squareView.layer.shadowRadius = ;
  33. self.squareView.layer.shadowColor = [UIColor redColor].CGColor;
  34. [self addSubview:self.squareView];
  35.  
  36. self.massSlider = [[UISlider alloc] initWithFrame:CGRectMake(, , , )];
  37. self.massSlider.minimumValue = ;
  38. self.massSlider.maximumValue = ;
  39. self.massSlider.tintColor = [UIColor cyanColor];
  40. self.massSlider.thumbTintColor = [UIColor redColor];
  41. self.massSlider.top = self.squareView.bottom + ;
  42. [self addSubview:self.massSlider];
  43. [self.massSlider addTarget:self action:@selector(durationChange:) forControlEvents:UIControlEventValueChanged];
  44.  
  45. self.stiffnessSlider = [[UISlider alloc] initWithFrame:CGRectMake(, , , )];
  46. self.stiffnessSlider.minimumValue = ;
  47. self.stiffnessSlider.maximumValue = ;
  48. self.stiffnessSlider.tintColor = [UIColor cyanColor];
  49. self.stiffnessSlider.thumbTintColor = [UIColor redColor];
  50. self.stiffnessSlider.top = self.massSlider.bottom + ;
  51. [self addSubview:self.stiffnessSlider];
  52. [self.stiffnessSlider addTarget:self action:@selector(durationChange:) forControlEvents:UIControlEventValueChanged];
  53.  
  54. self.dampingSlider = [[UISlider alloc] initWithFrame:CGRectMake(, , , )];
  55. self.dampingSlider.minimumValue = ;
  56. self.dampingSlider.maximumValue = ;
  57. self.dampingSlider.tintColor = [UIColor cyanColor];
  58. self.dampingSlider.thumbTintColor = [UIColor redColor];
  59. self.dampingSlider.top = self.stiffnessSlider.bottom + ;
  60. [self addSubview:self.dampingSlider];
  61. [self.dampingSlider addTarget:self action:@selector(durationChange:) forControlEvents:UIControlEventValueChanged];
  62.  
  63. self.massLabel = [[UILabel alloc] initWithFrame:CGRectMake(, , , )];
  64. self.massLabel.text = @"mass";
  65. self.massLabel.centerY = self.massSlider.centerY;
  66. self.massLabel.left = self.massSlider.right+;
  67. self.massLabel.textAlignment = NSTextAlignmentLeft;
  68. self.massLabel.textColor = [UIColor blackColor];
  69. self.massLabel.font = [UIFont systemFontOfSize:];
  70. [self addSubview:self.massLabel];
  71.  
  72. self.stiffnessLabel = [[UILabel alloc] initWithFrame:CGRectMake(, , , )];
  73. self.stiffnessLabel.text = @"stiffness";
  74. self.stiffnessLabel.centerY = self.stiffnessSlider.centerY;
  75. self.stiffnessLabel.left = self.stiffnessSlider.right+;
  76. self.stiffnessLabel.textAlignment = NSTextAlignmentLeft;
  77. self.stiffnessLabel.textColor = [UIColor blackColor];
  78. self.stiffnessLabel.font = [UIFont systemFontOfSize:];
  79. [self addSubview:self.stiffnessLabel];
  80.  
  81. self.dampingLabel = [[UILabel alloc] initWithFrame:CGRectMake(, , , )];
  82. self.dampingLabel.text = @"damping";
  83. self.dampingLabel.centerY = self.dampingSlider.centerY;
  84. self.dampingLabel.left = self.dampingSlider.right+;
  85. self.dampingLabel.textAlignment = NSTextAlignmentLeft;
  86. self.dampingLabel.textColor = [UIColor blackColor];
  87. self.dampingLabel.font = [UIFont systemFontOfSize:];
  88. [self addSubview:self.dampingLabel];
  89.  
  90. self.tableView = [[UITableView alloc] initWithFrame:CGRectMake(, self.dampingSlider.bottom+, SCREEN_WIDTH, SCREEN_HEIGHT-self.dampingSlider.bottom-) style:UITableViewStylePlain];
  91. self.tableView.delegate = self;
  92. self.tableView.dataSource = self;
  93. [self addSubview:self.tableView];
  94. }
  95.  
  96. /* The mass of the object attached to the end of the spring. Must be greater
  97. than 0. Defaults to one. */
  98.  
  99. //@property CGFloat mass;
  100.  
  101. /* The spring stiffness coefficient. Must be greater than 0.
  102. * Defaults to 100. */
  103.  
  104. //@property CGFloat stiffness;
  105.  
  106. /* The damping coefficient. Must be greater than or equal to 0.
  107. * Defaults to 10. */
  108.  
  109. //@property CGFloat damping;
  110.  
  111. /* The initial velocity of the object attached to the spring. Defaults
  112. * to zero, which represents an unmoving object. Negative values
  113. * represent the object moving away from the spring attachment point,
  114. * positive values represent the object moving towards the spring
  115. * attachment point. */
  116.  
  117. //@property CGFloat initialVelocity;
  118.  
  119. - (CASpringAnimation *)getAnimationKeyPath:(NSString *)keyPath fromValue:(id)fromValue toValue:(id)toValue{
  120. CASpringAnimation *springAnimation = [CASpringAnimation animationWithKeyPath:keyPath];
  121. springAnimation.fromValue = fromValue;
  122. springAnimation.toValue = toValue;
  123. springAnimation.mass = self.massSlider.value;
  124. springAnimation.stiffness = self.stiffnessSlider.value;
  125. springAnimation.damping = self.dampingSlider.value;
  126. springAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionLinear];;
  127. springAnimation.duration = ;
  128. springAnimation.repeatCount = ;
  129. /* animation remove from view after animation finish */
  130. springAnimation.removedOnCompletion = YES;
  131. /*
  132. 只有当removedOnCompletion设置为no时,fillmode设置为kCAFillModeBoth或者kCAFillModeForwards才有效,
  133. kCAFillModeRemoved //动画执行完成后回到初始状态
  134. kCAFillModeBackwards //动画执行完成后回到初始状态
  135. kCAFillModeForwards //动画执行完成后保留最后状态
  136. kCAFillModeBoth //动画执行完成后保留最后状态
  137. */
  138. springAnimation.fillMode = kCAFillModeForwards;
  139. /*
  140. 动画执行完成后按原动画返回执行,default no
  141. */
  142. // springAnimation.autoreverses = YES;
  143. return springAnimation;
  144. }
  145. - (void)initData{
  146. /*
  147. kCAScaleZ 缩放z 没有意义,因为是平面图形
  148. kCAPositionX设置y没有意义,可以随意设置,同理kCAPositionY设置x没有意义
  149. kCABackgroundColor,颜色变化必须要用CGColor
  150. 用到shadow的几个属性变化的时候,需要先设置shadow
  151. */
  152. NSValue *shadowStartPoint = [NSValue valueWithCGPoint:CGPointMake(, )];
  153. NSValue *shadowEndPoint = [NSValue valueWithCGPoint:CGPointMake(, )];
  154. id startColor = (id)([UIColor cyanColor].CGColor);
  155. id endColor = (id)([UIColor redColor].CGColor);
  156. id shadowStartColor = (id)[UIColor clearColor].CGColor;
  157. id shadowEndColor = (id)[UIColor redColor].CGColor;
  158. self.dataSource = [NSMutableArray array];
  159. NSArray *keypaths = @[kCARotation,kCARotationX,kCARotationY,kCARotationZ,
  160. kCAScale,kCAScaleX,kCAScaleZ,kCAPositionY,
  161. kCABoundsSizeW,kCAOpacity,kCABackgroundColor,kCACornerRadius,
  162. kCABorderWidth,kCAShadowColor,kCAShadowRadius,kCAShadowOffset];
  163.  
  164. NSArray *fromValues = @[@,@,@,@,
  165. @,@,@,@,
  166. @,@,startColor,@,
  167. @,shadowStartColor,@,shadowStartPoint];
  168.  
  169. NSArray *toValues = @[@(M_PI),@(M_PI),@(M_PI),@(M_PI),
  170. @,@,@,@,
  171. @,@,endColor,@,
  172. @,shadowEndColor,@,shadowEndPoint];
  173. for (int i=; i<keypaths.count; i++) {
  174. AnimationModel *model = [[AnimationModel alloc] init];
  175. model.keyPaths = keypaths[i];
  176. model.fromValue = fromValues[i];
  177. model.toValue = toValues[i];
  178. [self.dataSource addObject:model];
  179. }
  180. }
  181. - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
  182. return ;
  183. }
  184. - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
  185. return [self.dataSource count];
  186. }
  187. - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
  188. AnimationModel *model = [self.dataSource objectAtIndex:indexPath.row];
  189. UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"CellID"];
  190. if (cell==nil) {
  191. cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"CellID"];
  192. }
  193. cell.textLabel.text = model.keyPaths;
  194. cell.selectionStyle = ;
  195. return cell;
  196. }
  197. - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
  198. [self.squareView.layer removeAllAnimations];
  199. AnimationModel *model = [self.dataSource objectAtIndex:indexPath.row];
  200. CABasicAnimation *animation = [self getAnimationKeyPath:model.keyPaths fromValue:model.fromValue toValue:model.toValue];
  201. [self.squareView.layer addAnimation:animation forKey:@"animation"];
  202. }
  203. - (void)durationChange:(UISlider *)slider{
  204. [slider setValue:slider.value];
  205. if (slider==self.massSlider) {
  206. self.massLabel.text = [NSString stringWithFormat:@"mass %.2f", slider.value];
  207. }else if (slider==self.stiffnessSlider) {
  208. self.stiffnessLabel.text = [NSString stringWithFormat:@"stiffness %.2f", slider.value];
  209. }else if (slider==self.dampingSlider) {
  210. self.dampingLabel.text = [NSString stringWithFormat:@"damping %.2f", slider.value];
  211. }
  212. }
  213. @end
  214.  
  215. --------------------------------------------------------------------------------------
  216. #import <UIKit/UIKit.h>
  217.  
  218. @interface UIView (HPAdditions)
  219. @property (nonatomic) CGFloat top;
  220. @property (nonatomic) CGFloat bottom;
  221. @property (nonatomic) CGFloat left;
  222. @property (nonatomic) CGFloat right;
  223.  
  224. @property (nonatomic) CGFloat width;
  225. @property (nonatomic) CGFloat height;
  226. @property (nonatomic) CGPoint origin;
  227. @property (nonatomic) CGSize size;
  228.  
  229. @property (nonatomic) CGFloat centerX;
  230. @property (nonatomic) CGFloat centerY;
  231.  
  232. @end
  233.  
  234. #import "UIView+HPAdditions.h"
  235.  
  236. @implementation UIView (HPAdditions)
  237. - (CGFloat)top{
  238. return self.frame.origin.y;
  239. }
  240. - (void)setTop:(CGFloat)top{
  241. CGRect frame = self.frame;
  242. frame.origin.y = top;
  243. self.frame = frame;
  244. }
  245. - (CGFloat)bottom{
  246. return self.frame.origin.y+self.frame.size.height;
  247. }
  248. - (void)setBottom:(CGFloat)bottom{
  249. CGRect frame = self.frame;
  250. frame.origin.y = bottom-frame.size.height;
  251. self.frame = frame;
  252. }
  253. - (CGFloat)left{
  254. return self.frame.origin.x;
  255. }
  256. - (void)setLeft:(CGFloat)left{
  257. CGRect frame = self.frame;
  258. frame.origin.x = left;
  259. self.frame = frame;
  260. }
  261. - (CGFloat)right{
  262. return self.frame.origin.x+self.frame.size.width;
  263. }
  264. - (void)setRight:(CGFloat)right{
  265. CGRect frame = self.frame;
  266. frame.origin.x = right-frame.size.width;
  267. self.frame = frame;
  268. }
  269.  
  270. - (CGFloat)width{
  271. return self.frame.size.width;
  272. }
  273. - (void)setWidth:(CGFloat)width{
  274. CGRect frame = self.frame;
  275. frame.size.width = width;
  276. self.frame = frame;
  277. }
  278. - (CGFloat)height{
  279. return self.frame.size.height;
  280. }
  281. - (void)setHeight:(CGFloat)height{
  282. CGRect frame = self.frame;
  283. frame.size.height = height;
  284. self.frame = frame;
  285. }
  286. - (CGFloat)centerX{
  287. return self.center.x;
  288. }
  289. - (void)setCenterX:(CGFloat)centerX{
  290. self.center = CGPointMake(centerX, self.center.y);
  291. }
  292. - (CGFloat)centerY{
  293. return self.center.y;
  294. }
  295. - (void)setCenterY:(CGFloat)centerY{
  296. self.center = CGPointMake(self.center.x, centerY);
  297. }
  298. - (CGPoint)origin{
  299. return self.frame.origin;
  300. }
  301. - (void)setOrigin:(CGPoint)origin{
  302. CGRect frame = self.frame;
  303. frame.origin = origin;
  304. self.frame = frame;
  305. }
  306. - (CGSize)size{
  307. return self.frame.size;
  308. }
  309. - (void)setSize:(CGSize)size{
  310. CGRect frame = self.frame;
  311. frame.size = size;
  312. self.frame = frame;
  313. }
  314.  
  315. @end

  CABasicAnimation及其子类CASpringAnimation使用都是非常简单的,CASpringAnimation能够做出更加优雅的动画,建议使用。但是这两种动画能过制作的动画都是需要给定初始和最终状态值的,而中间过程需要系统去处理,这样的动画比较单一,下一篇将会介绍添加中间过程的动画CAKeyFrameAnimation关键帧动画,它能做出更加炫酷的动画。

iOS核心动画详解(CABasicAnimation)的更多相关文章

  1. iOS核心动画详解(一)

    前言 这篇文章主要是针对核心动画(Core Animation)的讲解,不涉及UIView的动画.因为内容较多,这篇文章会分为几个章节来进行介绍.本文主要是介绍核心动画的几个类之间的关系和CAAnim ...

  2. iOS应用开发详解

    <iOS应用开发详解> 基本信息 作者: 郭宏志    出版社:电子工业出版社 ISBN:9787121207075 上架时间:2013-6-28 出版日期:2013 年7月 开本:16开 ...

  3. iOS核心动画学习整理

    最近利用业余时间终于把iOS核心动画高级技巧(https://zsisme.gitbooks.io/ios-/content/chapter1/the-layer-tree.html)看完,对应其中一 ...

  4. IOS 核心动画之CAKeyframeAnimation - iBaby

    - IOS 核心动画之CAKeyframeAnimation - 简单介绍 是CApropertyAnimation的子类,跟CABasicAnimation的区别是:CABasicAnimation ...

  5. [转] ReactNative Animated动画详解

    http://web.jobbole.com/84962/     首页 所有文章 JavaScript HTML5 CSS 基础技术 前端职场 工具资源 更多频道▼ - 导航条 - 首页 所有文章 ...

  6. IOS 动画专题 --iOS核心动画

    iOS开发系列--让你的应用“动”起来 --iOS核心动画 概览 通过核心动画创建基础动画.关键帧动画.动画组.转场动画,如何通过UIView的装饰方法对这些动画操作进行简化等.在今天的文章里您可以看 ...

  7. iOS开发:详解Objective-C runTime

    Objective-C总Runtime的那点事儿(一)消息机制 最近在找工作,Objective-C中的Runtime是经常被问到的一个问题,几乎是面试大公司必问的一个问题.当然还有一些其他问题也几乎 ...

  8. iOS 核心动画 Core Animation浅谈

    代码地址如下:http://www.demodashi.com/demo/11603.html 前记 关于实现一个iOS动画,如果简单的,我们可以直接调用UIView的代码块来实现,虽然使用UIVie ...

  9. iOS核心动画高级技巧之核心动画(三)

    iOS核心动画高级技巧之CALayer(一) iOS核心动画高级技巧之图层变换和专用图层(二)iOS核心动画高级技巧之核心动画(三)iOS核心动画高级技巧之性能(四)iOS核心动画高级技巧之动画总结( ...

随机推荐

  1. JNI调用实例

    1. 环境 Windows7-64Bit VS2010-32Bit JDK1.8-64Bit 2. 步骤 2.1 创建NativePrint类 public class NativePrint { p ...

  2. 常见typedef 用法

    1.常规变量类型定义例如:typedef unsigned char uchar描述:uchar等价于unsigned char类型定义      uchar c声明等于unsigned char c ...

  3. config.sql

    # mysql服务器注释支持# #到该行结束# -- 到该行结束 # /* 行中间或多个行 */ drop database if exists db_warehouse;create databas ...

  4. redis命令_ZREVRANGEBYSCORE

    ZREVRANGEBYSCORE key max min [WITHSCORES] [LIMIT offset count] 返回有序集 key 中, score 值介于 max 和 min 之间(默 ...

  5. 0054 Spring MVC的@Controller和@RequestMapping注解

    @Controller注解 该注解用来指示一个类是一个控制器,在Spring的配置xml文件中开启注解扫描 <context:conponent-scan base-package=" ...

  6. iOS 转盘抽奖游戏(原生)

    转盘抽奖游戏在一般的app中都会有,应该算是一种吸引用户的一种手段.在项目中集成转盘抽奖游戏,大都采用h5的方式来实现,但是由于项目需求,需要在app中使用原生来实现转盘抽奖.实现原理也很简单,中间的 ...

  7. PHP——菜单及内容轮换(Jquery)

    效果: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3. ...

  8. Yii2 Api认证和授权(翻译)

    Authentication 认证 RESTful Api 是无状态的, 因此这意味着不能使用 sessions && cookies. 因此每一个请求应该带有一些 authentic ...

  9. hdu3667 Transportation 费用与流量平方成正比的最小流 拆边法+最小费用最大流

    /** 题目:hdu3667 Transportation 拆边法+最小费用最大流 链接:http://acm.hdu.edu.cn/showproblem.php?pid=3667 题意:n个城市由 ...

  10. IO模型(阻塞、非阻塞、多路复用与异步)

    IO模型 同步IO和异步IO,阻塞IO和非阻塞IO分别是什么,到底有什么区别?不同环境下给出的答案也是不一的.所以先限定一下上下文是非常有必要的. 本文讨论的背景是Linux环境下的network I ...