CMPedometer(CoreMotion)实时更新

Rob*_*hai 5 objective-c ios core-motion

我发现有关此主题(CMPedometer)的资源非常有限。我想知道这里是否有人设法使它正常工作。我的代码非常简单,并且具有比我想做的更多的功能。基本上,步数计数器不会使用户执行的每一步都递增。

它实际上是在跟踪用户执行的每个步骤,但更新速度如此之慢,我不知道为什么。我什至尝试使用NSTimer发出每半秒更新标签的请求。我想尝试让步数计数器在用户迈出一步时进行更新。这是我的代码...

#import "ViewController.h"
#import <CoreMotion/CoreMotion.h>

@interface ViewController ()

@property (nonatomic, strong) CMPedometer    *pedometer;
@property (nonatomic, weak) IBOutlet UILabel *startDateLabel;
@property (nonatomic, weak) IBOutlet UILabel *endDateLabel;
@property (nonatomic, weak) IBOutlet UILabel *stepsLabel;
@property (nonatomic, weak) IBOutlet UILabel *distanceLabel;
@property (nonatomic, weak) IBOutlet UILabel *ascendedLabel;
@property (nonatomic, weak) IBOutlet UILabel *descendedLabel;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    if ([CMPedometer isStepCountingAvailable]) {
        self.pedometer = [[CMPedometer alloc] init];
        [NSTimer scheduledTimerWithTimeInterval:0.5f
                                         target:self
                                       selector:@selector(recursiveQuery)
                                       userInfo:nil
                                        repeats:YES];
    } else {
        NSLog(@"Nothing available");
        self.startDateLabel.text = @"";
        self.endDateLabel.text   = @"";
        self.stepsLabel.text     = @"";
        self.distanceLabel.text  = @"";
        self.ascendedLabel.text  = @"";
        self.descendedLabel.text = @"";
    }

}

- (void)viewWillAppear:(BOOL)animated {
    [super viewWillAppear:animated];

    [self.pedometer startPedometerUpdatesFromDate:[NSDate date]
                                      withHandler:^(CMPedometerData *pedometerData, NSError *error) {
                                          dispatch_async(dispatch_get_main_queue(), ^{
                                              NSLog(@"data:%@, error:%@", pedometerData, error);
                                          });
                                      }];
}


- (void)viewWillDisappear:(BOOL)animated {
    [super viewWillDisappear:animated];
    [self.pedometer stopPedometerUpdates];
}

- (NSString *)stringWithObject:(id)obj {
    return [NSString stringWithFormat:@"%@", obj];
}

- (NSString *)stringForDate:(NSDate *)date {

    NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
    formatter.dateStyle = NSDateFormatterShortStyle;
    formatter.timeStyle = NSDateFormatterShortStyle;

    return [formatter stringFromDate:date];
}

- (void)queryDataFrom:(NSDate *)startDate toDate:(NSDate *)endDate {
    [self.pedometer queryPedometerDataFromDate:startDate
                                        toDate:endDate
                                   withHandler:
     ^(CMPedometerData *pedometerData, NSError *error) {

         NSLog(@"data:%@, error:%@", pedometerData, error);

         dispatch_async(dispatch_get_main_queue(), ^{
             if (error) {
                 NSLog(@"Error = %@",error.userInfo);
                 self.startDateLabel.text = @"";
                 self.endDateLabel.text   = @"";
                 self.stepsLabel.text     = @"";
                 self.distanceLabel.text  = @"";
                 self.ascendedLabel.text  = @"";
                 self.descendedLabel.text = @"";
             } else {
                 self.startDateLabel.text = [self stringForDate:pedometerData.startDate];
                 self.endDateLabel.text   = [self stringForDate:pedometerData.endDate];
                 self.stepsLabel.text     = [self stringWithObject:pedometerData.numberOfSteps];
                 self.distanceLabel.text  = [NSString stringWithFormat:@"%.1f[m]", [pedometerData.distance floatValue]];
                 self.ascendedLabel.text  = [self stringWithObject:pedometerData.floorsAscended];
                 self.descendedLabel.text = [self stringWithObject:pedometerData.floorsDescended];
             }
         });
     }];
}

- (void)recursiveQuery {
    NSDate *to   = [NSDate date];
    NSDate *from = [to dateByAddingTimeInterval:-(24. * 3600.)];
    [self queryDataFrom:from toDate:to];
}
Run Code Online (Sandbox Code Playgroud)

预先感谢您的任何反馈!

编辑

看来,用于实时更新的适当方法如下。

- (void)liveSteps {
    [self.pedometer startPedometerUpdatesFromDate:[NSDate date]
                                      withHandler:^(CMPedometerData *pedometerData, NSError *error) {
                                          dispatch_async(dispatch_get_main_queue(), ^{
                                              NSLog(@"Steps %@",pedometerData.numberOfSteps);
                                          });
                                      }];
}
Run Code Online (Sandbox Code Playgroud)

但是,即使这样严重延迟。有谁知道如何适当地使用它来在用户迈出一步时进行更新?

dei*_*ter 4

我只能证实你的发现。我也想获得“真实”的实时信息。目前看来,API 无法做到这一点;甚至通过强制更新进入队列、同步、异步等。

对于有此问题的参考和其他人,这里是我基于 Swift 3 和 Xcode 8.2 使用的代码。在检查 CMPedometer.isStepCountingAvailable() 之后,我只需将这部分代码应用到相关视图控制器中。

如您所见,我添加了一个小动画,以便以更流畅的方式更新 UILabel。

    // Steps update in near realtime - UILabel
    self.pedoMeter.startUpdates(from: midnightOfToday) { (data: CMPedometerData?, error) -> Void in

        DispatchQueue.main.async(execute: { () -> Void in
            if(error == nil){
                self.todaySteps.text = "\(data!.numberOfSteps)"
                // Animate the changes of numbers in the UILabel
                UILabel.transition(with: self.todaySteps,
                                   duration: 0.50,
                                   options: .transitionCrossDissolve,
                                   animations: nil,
                                   completion: nil)
            }
        })
    }
Run Code Online (Sandbox Code Playgroud)