Animate UILabel text between two numbers?
Asked Answered
S

6

81

I'm new to iPhone and Mac programming (developed for Windows before), and I've got a question:

How do I animate the text property of an UILabel between two numbers, e.g. from 5 to 80 in an Ease-Out style? Is it possible with CoreAnimation? I have been searching on Google for an hour, but I haven't found anything solving my problem. What I want: Animate the users money for a simple game. It doesn't look very nice when it just goes from 50 to 100 or something like that without animation.

Anyone having an idea how to do that?

Thanks!

Schroer answered 14/3, 2011 at 16:15 Comment(2)
This is an old question but Pop (Facebook animation framework) would be a good solution. You could animate the text property directly.Karns
Possible duplicate of Animate text change in UILabelMcreynolds
M
160

You can use the automatic transitions. It's working perfectly well :

// Add transition (must be called after myLabel has been displayed)
 CATransition *animation = [CATransition animation];
animation.duration = 1.0;
animation.type = kCATransitionFade;
animation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];
[myLabel.layer addAnimation:animation forKey:@"changeTextTransition"];

// Change the text
myLabel.text = newText;

This code works if myLabel is already displayed. If not myLabel.layer will be nil and the animation will not be added to the object.


in Swift 4 that would be:

let animation: CATransition = CATransition()
animation.duration = 1.0
animation.type = kCATransitionFade
animation.timingFunction = CAMediaTimingFunction(name: kCAMediaTimingFunctionEaseInEaseOut)
myLabel.layer.add(animation, forKey: "changeTextTransition")
Moral answered 7/6, 2011 at 15:3 Comment(8)
I tried this (ensuring that the layer is not nil when the animation is added), but the text still doesn't animate. Any other gotchas?Brevity
Just a side note, you have to add the animation right before you change the text each time you want to change the text and have the fade effect happen. You don't however have to instantiate a new Transition each time, the same instance can be reused over and over.Contiguity
The question was how to animate the numbers e.g. incrementing them in value, not graphically fading between the two. Use PRTween: github.com/shu223/TweenDemo/tree/…Surprint
Actually, if you set removedOnCompletion to NO on the instance of CATransition, every change to the label's text will be animated.Chocolate
didn't work for me: I use this code when change text from slider value - no luck =(Disadvantageous
How do you even know the key for this? I mean there are some common keys for layer property like opacity, position, etc. But where do you find the changeTextTransition?? I can't find any document for this.Manyplies
@Manyplies No idea where a doc could be and if there is any. Found that key on a tutorial years ago.Moral
@Manyplies There's nothing special about the key changeTextTransition. It's simply a string that identifies the animation, which means it's a string that you can use later to identify an animation that you previously added. If you don't need to be able to identify your animation later, you can just specify nil.Gamboa
V
27

It works well!

Objective-C

[UIView transitionWithView:self.label 
                  duration:.5f 
                   options:UIViewAnimationOptionCurveEaseInOut | 
                           UIViewAnimationOptionTransitionCrossDissolve 
                animations:^{

    self.label.text = rand() % 2 ? @"111!" : @"42";

} completion:nil];

Swift 2

UIView.transitionWithView(label, duration: 0.25, options: [.CurveEaseInOut, .TransitionCrossDissolve], animations: {
    self.label.text = (arc4random() % 2 == 0) ? "111" : "222"
}, completion: nil)

Swift 3, 4, 5

UIView.transition(with: label, duration: 0.25, options: [.curveEaseInOut, .transitionCrossDissolve], animations: {
    self.label.text = (arc4random() % 2 == 0) ? "111" : "222"
}, completion: nil)
Verniavernice answered 17/9, 2014 at 12:49 Comment(1)
Worked also for other properties, such as changing the image of a UIImageViewCyclostome
S
7

I found a great engine for tweening values with a variety of different timing functions called PRTween. Install the classes and create some code along these lines:

- (IBAction)tweenValue
{
    [[PRTween sharedInstance] removeTweenOperation:activeTweenOperation];
    PRTweenPeriod *period = [PRTweenPeriod periodWithStartValue:0.0 endValue:100.0 duration:1.0];
    activeTweenOperation = [[PRTween sharedInstance] addTweenPeriod:period
                                                             target:self
                                                           selector:@selector(update:)
                                                     timingFunction:&PRTweenTimingFunctionCircOut];
}

- (void)update:(PRTweenPeriod*)period
{
    self.animatingView.center = CGPointMake(period.tweenedValue + 100.0, 200.0);
    self.valueLabel.text = [NSString stringWithFormat:@"%.2f", period.tweenedValue];
}

Works a treat for me. :)

Surprint answered 11/6, 2013 at 15:23 Comment(0)
I
5

If you kind of want it to count up and down with the new number pushing the previous number away (like a ticker or something):

let animation = CATransition()
animation.removedOnCompletion = true
animation.duration = 0.2
animation.type = kCATransitionPush
animation.subtype = newValue > value ? kCATransitionFromTop : kCATransitionFromBottom
animation.timingFunction = CAMediaTimingFunction(name: kCAMediaTimingFunctionEaseOut)
valueLabel.layer.addAnimation(animation, forKey:"changeTextTransition")
Inveteracy answered 17/11, 2015 at 17:5 Comment(2)
Hey! Thanks for your answer! This works pretty neat. I have only on further question. Do you know how to animate only parts of the text? Like the 3rd character and let the rest be unchanged? I would like to implement a label that shows numbers. If the value changes only the digits shall animate, that actually changed.Seedman
Personally I'd look into using CoreGraphics and CoreAnimation for this kind of thing. Or you could cheat and string a couple of labels together? Depends on how complex you string is.Inveteracy
U
3

In Swift 2.0, using the UIView.transitionWithView() method:

UIView.transitionWithView(self.payPeriodSummaryLabel,
        duration: 0.2,
        options: [.CurveEaseInOut, .TransitionCrossDissolve],
        animations: { () -> Void in
            self.label.text = "your text value"
        }, completion: nil)
Underquote answered 25/1, 2016 at 20:25 Comment(0)
J
1

another simple alternative

extension UILabel {    
    func countAnimation(upto: Double) {
        let from: Double = text?.replace(string: ",", replacement: ".").components(separatedBy: CharacterSet.init(charactersIn: "-0123456789.").inverted).first.flatMap { Double($0) } ?? 0.0
        let steps: Int = 20
        let duration = 0.350
        let rate = duration / Double(steps)
        let diff = upto - from
        for i in 0...steps {
            DispatchQueue.main.asyncAfter(deadline: .now() + rate * Double(i)) {
                self.text = "\(from + diff * (Double(i) / Double(steps)))"
            }
        }
    }
}
Jitney answered 13/5, 2020 at 11:48 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.