使用UIBezierPath的自定义计时器动画

时间:2014-01-15 00:16:11

标签: ios objective-c nstimer countdown uibezierpath

我目前有一个计时器,低于该计时器动画一条线 - 一条倒计时线。这会根据我传递给_percent的秒数来设置动画。

我想要另一个描绘秒数的圆形轮。然而,我正在努力研究如何在_percent中每秒留下1秒的动画。例如如果百分比是100,我想要圆圈动画一个循环100次持续1秒。 (以及实现下面显示的动画效果。

这是为了清晰起见的图:)第一行的3显示了当前的情况,底部的一行显示了我想要的效果...... enter image description here

以下是我在视图控制器类中启动计时器的方法:

-(void)viewDidLoad
{
       CPCircleCountdownView *m_testView;
    m_testView.percent = 100;
}

- (IBAction)StartTimerButtonPressed:(id)sender
{
    // Kick off a timer to count it down
    m_timer = [NSTimer scheduledTimerWithTimeInterval:0.1 target:self selector:@selector(decrementSpin) userInfo:nil repeats:YES];
}

- (void)decrementSpin
{
    // If we can decrement our percentage, do so, and redraw the view
    if (m_testView.percent > 0) {
        m_testView.percent = m_testView.percent - 1;
        [m_testView setNeedsDisplay];
    }
    else {
        [m_timer invalidate];
        m_timer = nil;
    }
}

这是我的CPCountdownView,我想在每秒左右添加持续一秒的新圆形动画。理论上的两种动画应该同时完成。

#import "CPCircleCountdownView.h"

@interface CPCircleCountdownView ()
{
    CGFloat startAngle;
    CGFloat endAngle;
}

@end

@implementation CPCircleCountdownView

- (id)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
        // Initialization code

        self.backgroundColor = [UIColor clearColor];

        //get start and end angles
        startAngle = M_PI * 1.5;
        endAngle = startAngle + (M_PI * 2);
    }
    return self;
}

- (void)drawRect:(CGRect)rect
{
    //Display arc percent
    //NSString *textContent = [NSString stringWithFormat:@"%d", self.percent];    //TODO: Pass total number of seconds int

    //-------------------------
    UIBezierPath* bezierPath = [UIBezierPath bezierPath];
    // Create our arc, with the correct angles
    [bezierPath addArcWithCenter:CGPointMake(rect.size.width / 2, rect.size.height / 2)
                          radius:130
                      startAngle:startAngle
                        endAngle:(endAngle - startAngle) * (_percent / _overallScore) + startAngle
                       clockwise:YES];
    // Set the display for the path, and stroke it
    bezierPath.lineWidth = 2;
    [[UIColor colorWithRed:122.0 / 255.0 green:197.0 / 255.0 blue:205.0  / 255.0 alpha:1.0] setStroke];
    [bezierPath stroke];

    //-------------------------  
}

任何帮助都会很棒我是比较新的所以如果你能给我一些很棒的例子:)谢谢

1 个答案:

答案 0 :(得分:0)

可能更好的方法是使用2个CAShapeLayers并为bezier路径设置动画。假设CPCircleCountdownView是一个UIView子类

 @interface CPCircleCountdownView ()

  @property (nonatomic) CAShapeLayer* layer1;
  @property (nonatomic) CAShapeLayer* layer2;
  @property (nonatomic, assign) float startAngle;
  @property (nonatomic, assign) float endAngle;    
 @end   

 @implementation CPCircleCountdownView

 - (void)intiWithFrame:(CGRect) Frame
 {
   self = [super initWithFrame:frame];
   if (self) {
      _startAngle = M_PI * 1.5;
     _endAngle = _startAngle + (M_PI * 2);

    }


 }

-(void)layoutSubviews{

    self.layer1 = [CAShapeLayer layer];
    self.layer1.fillColor = [UIColor clearColor].CGColor;
    self.layer1.strokeColor = [UIColor redColor].CGColor;

    self.layer2 = [CAShapeLayer layer];
    self.layer2.fillColor = [UIColor clearColor].CGColor;
    self.layer2.strokeColor = [UIColor greenColor].CGColor;

    [self.layer1 setFrame:CGRectMake(50, 50, 200, 200)];
    [self.layer2 setFrame:CGRectMake(50, 50, 200, 200)];
    UIBezierPath *path1 = [UIBezierPath bezierPathWithArcCenter:CGPointMake(self.layer1.frame.size.width / 2, self.layer1.frame.size.height / 2)
                                                     radius:130
                                                 startAngle:self.startAngle
                                                   endAngle:self.endAngle
                                                  clockwise:YES];

    UIBezierPath *path2 = [UIBezierPath bezierPathWithArcCenter:CGPointMake(self.layer2.frame.size.width / 2, self.layer2.frame.size.height / 2)
                                                     radius:125
                                                 startAngle:self.endAngle
                                                   endAngle:self.startAngle
                                                  clockwise:NO];

    self.layer1.path = path1.CGPath;
    self.layer2.path = path2.CGPath;
    [self.view.layer addSublayer:self.layer1];
    [self.view.layer addSublayer:self.layer2];
    [self.layer1  setStrokeEnd:1.0f];
    [self.layer2 setStrokeEnd:0.0f];
   // i've placed this here for convenience in testing, really it should be in method called by your button Action
    [self decrementSpinForValue:100];

}

- (void)animationDidStop:(CAAnimation *)animation finished:(BOOL)finished {
    [self.layer1 removeAllAnimations];
    [self.layer2 removeAllAnimations];
}


-(void) decrementSpinForValue:(float) percent{

    CABasicAnimation *strokeEndAnimation = [CABasicAnimation animationWithKeyPath:@"strokeEnd"];
    strokeEndAnimation.delegate = self;
    strokeEndAnimation.duration = percent;
    [strokeEndAnimation setFillMode:kCAFillModeForwards];
    strokeEndAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionLinear];
    strokeEndAnimation.removedOnCompletion = NO;
    strokeEndAnimation.fromValue = [NSNumber numberWithFloat:1.0f];
    strokeEndAnimation.toValue = [NSNumber numberWithFloat:0.0f];
    [self.layer1 addAnimation:strokeEndAnimation forKey:@"progressOuterStatus"];

    CABasicAnimation *strokeEndAnimation2 = [CABasicAnimation animationWithKeyPath:@"strokeEnd"];
    strokeEndAnimation2.duration = 1.0f;
    strokeEndAnimation2.repeatCount = percent;
    [strokeEndAnimation2 setFillMode:kCAFillModeForwards];
    strokeEndAnimation2.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionLinear];
    strokeEndAnimation2.removedOnCompletion = NO;
    strokeEndAnimation2.fromValue = [NSNumber numberWithFloat:0.0f];
    strokeEndAnimation2.toValue = [NSNumber numberWithFloat:1.0f];
    [self.layer2 addAnimation:strokeEndAnimation2 forKey:@"progressInnerStatus"];


}

@end

我认为这可以让您了解如何实现目标。