How can I programmatically pause an NSTimer?

前端 未结 15 1632
梦毁少年i
梦毁少年i 2020-11-27 13:06

I\'m using an NSTimer to do some rendering in an OpenGL based iPhone app. I have a modal dialog box that pops up and requests user input. While the user is providing input

相关标签:
15条回答
  • 2020-11-27 13:31

    If your fire time interval is tolerant, you can use CADisplayLink instead of NSTimer. Because CADisplayLink support -pause.

    displayLink = [[CADisplayLink displayLinkWithTarget:self selector:@selector(fireMethod:)];
    displayLink.frameInterval = approximateVaue;//CADisplayLink's frame rate is 60 fps.
        [displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
    
    //pause
    [displayLink pause];
    
    0 讨论(0)
  • 2020-11-27 13:38

    The way I accomplished this was by storing the NSTimer's firing date in a variable and then setting the firing date to INFINITY.

    When I pause:

    pauseStart = [[NSDate dateWithTimeIntervalSinceNow:0] retain];
    previousFiringDate = [timer firingDate];
    [timer setFiringDate:INFINITY]
    

    When I unpause I add the amount of time that the timer was paused to the previous firing date:

    float pauseTime = -1*[pauseStart timeIntervalSinceNow];
    [timer setFireDate:[previousFireDate initWithTimeInterval:pauseTime sinceDate:previousFireDate]];
    

    This made it a lot easier than worrying about invalidating and reinitializing the timer. I had a lot of timers so I just put them all in an array and did this to all of them. I subclassed the NSTimer in order to keep the previousFiringDate associated with that timer.

    This was also better than using a BOOL of whether or not it was paused so actions wouldn't take place if the BOOL was set. This is because if something was about to happen before you pause, it won't happen after you unpause it because it will have just been skipped.

    0 讨论(0)
  • 2020-11-27 13:40

    Here is a category on NSTimer that allows pause and resume functionality.

    Header:

    #import <Foundation/Foundation.h>
    
    @interface NSTimer (CVPausable)
    
    - (void)pauseOrResume;
    - (BOOL)isPaused;
    
    @end
    

    Implementation:

    #import "NSTimer+CVPausable.h"
    #import <objc/runtime.h>
    
    @interface NSTimer (CVPausablePrivate)
    
    @property (nonatomic) NSNumber *timeDeltaNumber;
    
    @end
    
    @implementation NSTimer (CVPausablePrivate)
    
    static void *AssociationKey;
    
    - (NSNumber *)timeDeltaNumber
    {
        return objc_getAssociatedObject(self, AssociationKey);
    }
    
    - (void)setTimeDeltaNumber:(NSNumber *)timeDeltaNumber
    {
        objc_setAssociatedObject(self, AssociationKey, timeDeltaNumber, OBJC_ASSOCIATION_RETAIN);
    }
    
    @end
    
    
    @implementation NSTimer (CVPausable)
    
    - (void)pauseOrResume
    {
        if ([self isPaused]) {
            self.fireDate = [[NSDate date] dateByAddingTimeInterval:[self.timeDeltaNumber doubleValue]];
            self.timeDeltaNumber = nil;
        }
        else {
            NSTimeInterval interval = [[self fireDate] timeIntervalSinceNow];
            self.timeDeltaNumber = @(interval);
            self.fireDate = [NSDate distantFuture];
        }
    }
    
    - (BOOL)isPaused
    {
        return (self.timeDeltaNumber != nil);
    }
    
    @end
    
    0 讨论(0)
提交回复
热议问题