我正在尝试开发一个使用 CoreMotion 框架的应用程序。我不想一直跑出去测试我的应用程序,所以我想找到一种技术,让我可以发送假的,这样我就可以尝试很多不同的场景。我希望测试用户步行、驾驶、停车等场景。我不想每次我想测试应用程序时都必须出去走路、开车、停车等。
编辑:我知道模拟器还不支持 CoreMotion。但希望有人能给我一些关于如何伪造这个的想法
更新:到目前为止我使用方法调配所取得的成就。我已经为此换回了objective-c,因为我认为最好先尝试用这种语言弄清楚,然后再尝试将其翻译成Swift
我已经像这样设置了一个方法调配类
+ (void)swizzleClass:(Class)class method:(NSString*)methodName {
SEL originalSelector = NSSelectorFromString(methodName);
SEL newSelector = NSSelectorFromString([NSString stringWithFormat:@"%@%@", @"override_", methodName]);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method newMethod = class_getInstanceMethod(class, newSelector);
method_exchangeImplementations(originalMethod, newMethod);
}
我为 CMMotion 和 CMAccelerometerData 创建了一个类别
- (void) simx_setAcceleration:(CMAcceleration)acceleration {
NSValue *value = [NSValue value:&acceleration withObjCType:@encode(CMAcceleration)];
objc_setAssociatedObject(self, ACCELERATION_IDENTIFIER, value, OBJC_ASSOCIATION_RETAIN);
}
- (CMAcceleration)override_acceleration {
NSValue *value = objc_getAssociatedObject(self, ACCELERATION_IDENTIFIER);
CMAcceleration acc;
[value getValue:&acc];
return acc;
}
然后是 CMMotionManager 类的类别。
- (void) simx_setAccelerationData:(CMAcceleration )accelerationData
{
NSValue *value = [NSValue value:&accelerationData withObjCType:@encode(CMAcceleration)];
objc_setAssociatedObject(self, HANDLER_IDENTIFIER, value, OBJC_ASSOCIATION_RETAIN);
}
- (CMAccelerometerData *)override_accelerometerData
{
NSValue *value = objc_getAssociatedObject(self, HANDLER_IDENTIFIER);
CMAcceleration acc;
[value getValue:&acc];
CMAccelerometerData *data = [[CMAccelerometerData alloc] init];
//Add
[data simx_setAcceleration:acc];
return data;
}
Swizzling方法是这样完成的
[CESwizzleUtils swizzleClass:[CMMotionManager class]
method:@"INSERT METHOD NAME TO BE SWIZZLED HERE"];
这允许我传入我自己的数据
//Add
CMAccelerometerData *data = [[CMAccelerometerData alloc] init];
[data simx_setAcceleration:acc];
[motionManager simx_setAccelerationData:acc];
所以我可以像这样检索数据
motionManager.accelerometerData.acceleration.x;
我也对 DeviceMotion 类进行了调配。这是我拥有的快速示例应用程序,它使用方法 swizzle 技术从加速度计和陀螺仪中提取数据
单击测试按钮时,它会随机生成加速度计和陀螺仪数据并更新标签。
代码看起来像这样
-(IBAction)testing:(id)sender
{
//random double just generates a random double between 0 and 1
CMAcceleration acc;
acc.x = [self randomDouble:0 :1];
acc.y = [self randomDouble:0 :1];
acc.z = [self randomDouble:0 :1];
//Add
CMAccelerometerData *data = [[CMAccelerometerData alloc] init];
[data simx_setAcceleration:acc];
[motionManager simx_setAccelerationData:acc];
//Sim gravity and userAcel
CMAcceleration grav;
grav.x = [self randomDouble:0 :1];
grav.y = [self randomDouble:0 :1];
grav.z = [self randomDouble:0 :1];
CMAcceleration userAcel;
userAcel.x = [self randomDouble:0 :1];
userAcel.y = [self randomDouble:0 :1];
userAcel.z = [self randomDouble:0 :1];
CMDeviceMotion *deviceMotion = [[CMDeviceMotion alloc] init];
[deviceMotion simx_setUserAcceleration:userAcel];
[deviceMotion simx_setGravity:grav];
[motionManager simx_setDeviceMotion:deviceMotion];
accelLabael.text = [NSString stringWithFormat:@"Accelerometer: %.02f %.02f %.02f", motionManager.accelerometerData.acceleration.x,motionManager.accelerometerData.acceleration.y,motionManager.accelerometerData.acceleration.z];
gravityLabel.text = [NSString stringWithFormat:@"Gravity: %.02f %.02f %.02f", motionManager.deviceMotion.gravity.x,motionManager.deviceMotion.gravity.y,motionManager.deviceMotion.gravity.z];
accelSpeedLabel.text = [NSString stringWithFormat:@"Accel: %.02f %.02f %.02f", motionManager.deviceMotion.userAcceleration.x,motionManager.deviceMotion.userAcceleration.y,motionManager.deviceMotion.userAcceleration.z];
}
我正在努力弄清楚如何获得这种方法。我已经混合了许多 CoreMotion 方法,但需要一些帮助。目前在模拟器上,尽管 MotionManager 现在存储陀螺仪、加速度计等数据,但这些块方法不会触发。
[activityManager startActivityUpdatesToQueue:[[NSOperationQueue alloc] init] withHandler: ^(CMMotionActivity *activity){
}];
[motionManager startDeviceMotionUpdatesToQueue:[[NSOperationQueue alloc] init]
withHandler:^ (CMDeviceMotion *motionData, NSError *error) {
}];
我也混合了这些方法
- (BOOL)override_isAccelerometerActive;
- (BOOL)override_isDeviceMotionActive;
-(BOOL)override_isGyroAvailable;
所以他们总是返回真,但仍然不能让这些块触发。我想要一些帮助,试图弄清楚如何正确调整这些方法,以便我可以开始向模拟器发送和接收模拟数据
编辑:我确实通过添加以下类别来调整加速度计更新方法。到 CMMotionManager 类
-(void)override_startAccelerometerUpdatesToQueue:(NSOperationQueue *)queue
withHandler:(CMAccelerometerHandler)handler{
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"Test %.02f %.02f %.02f", self.accelerometerData.acceleration.x,self.accelerometerData.acceleration.y,self.accelerometerData.acceleration.z);
});
}
这工作得很好,因为我已经混合了加速度数据。
然后我通过添加此类别尝试使用 CMMotionActivityManager 类
-(void)override_startActivityUpdatesToQueue:(NSOperationQueue *)queue
withHandler:(CMMotionActivity *)activity
{
dispatch_async(dispatch_get_main_queue(), ^
{
BOOL walking = [activity walking];
NSLog(@"%i", walking);
});
}
但是我在这里收到此错误
[ NSGlobalBlock行走]:无法识别的选择器发送到实例 0x1086cf330
任何建议,将不胜感激
更新 2:
抱歉回复晚了,只能等到今天试试这个。我根据您的建议更新了方法,因此现在可以使用
-(void)override_startActivityUpdatesToQueue:(NSOperationQueue *)queue
withHandler:(CMMotionActivityHandler )activity
{
}
问题是,我需要访问 CMMotionActivity 才能确定他们是否在走路等。原始代码
[activityManager startActivityUpdatesToQueue:[[NSOperationQueue alloc] init] withHandler: ^(CMMotionActivity *activity){
dispatch_async(dispatch_get_main_queue(), ^
{
[activity walking];
});
}];
允许您访问此变量。但是,既然我已经对此进行了调整,它现在在我的类别文件中调用我的新声明,其中不包含 CMMotionActivity 变量。关于如何访问它的任何想法。这有点复杂,但这是我最终开始模拟 CoreMotion 数据之前的最后一个障碍。我已经模拟了陀螺仪和指南针的数据,并且我已经从真实的旅程中获得了数据,所以我可以将其输入模拟器。但是我需要让它告诉我用户是否在走路、跑步、开车等。
有什么建议么?