MotionBegan: не работает

Я сталкиваюсь с проблемой, когда пытаюсь использовать (void) motionBegan: (UIEventSubtype) motion withEvent: (UIEvent *) событие, чтобы захватить событие shake. Проблема в том, что функция даже не работает, даже когда я переопределяю canBecomeFirstResponder и устанавливаю ее для возврата YES. Я видел некоторые другие сообщения с этой проблемой, но я не нашел ответа.

Спасибо за любую помощь!

Первый пример .h(класс, унаследованный от UIView - "вызван" из класса делегата приложения) {

@class TestApplicationView;

@interface TestApplicationView : UIView {

    IBOutlet UIView *view;
}

}

Первый пример .m {

- (id)initWithCoder:(NSCoder *)coder
{
    [self setUpView];
    return self;
}

- (id)initWithFrame:(CGRect)frame
{
    [self setUpView];
    return self;
}

- (void)setUpView
{
    [self becomeFirstResponder];
    NSLog(@"First Responder - %d", [self isFirstResponder]);
}

}

Второй пример .h(класс унаследован от UIApplicationDelegate и UIScrollViewDelegate) {

#import <UIKit/UIKit.h>

@class TestApplicationViewController;

@interface TestApplicationAppDelegate : NSObject <UIApplicationDelegate, UIScrollViewDelegate> {

IBOutlet UIWindow *window;
IBOutlet UIScrollView *scrollView;
}

}

Второй пример .m {

- (void)applicationDidFinishLaunching:(UIApplication *)application
{
    [self becomeFirstResponder];
}

}

- Второй пример возвращает следующее предупреждение: "TestApplicationAppDelegate" может не отвечать на "-becomeFirstResponder"

Ответ 1

Я предполагаю, что вы хотите реализовать это в подклассе UIViewController. Сделайте UIViewController способным стать первым ответчиком:

- (BOOL)canBecomeFirstResponder {
     return YES;
}

Сделайте UIViewController первым ответчиком в viewDidAppear:.

- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    [self becomeFirstResponder];
}

Если в представлении содержится любой элемент, такой как UITextField, который может стать первым ответчиком, убедитесь, что в какой-то момент элемент уходит в отставку. Например, при UITextField UIViewController необходимо реализовать протокол UITextFieldDelegate и фактически быть делегатом. Тогда в textFieldShouldReturn:

- (BOOL)textFieldShouldReturn:(UITextField *)theTextField {
    // Hides the keyboard
    [theTextField resignFirstResponder];
    // Returns first responder status to self so that shake events register here
    [self becomeFirstResponder];
    return YES;
}

Внедрить motionEnded:withEvent:

- (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event {
    if ( event.subtype == UIEventSubtypeMotionShake ) {
    // Do something
    }

    if ([super respondsToSelector:@selector(motionEnded:withEvent:)]) {
        [super motionEnded:motion withEvent:event];
    }
}

В iPhone Developer Forums есть хороший пост от Matt Drance из Apple (требуется регистрация в качестве разработчика).

Обновление: реализация в подклассе UIView

Как обсуждалось в комментариях, это также не удивительно, в подклассе UIView. Вот как построить минимальный пример.

Создайте новый проект приложения на основе View, назовите его ShakeTest. Создайте новый подкласс UIView, назовите его ShakeView. Сделайте ShakeView.h следующим образом:

#import <UIKit/UIKit.h>
@interface ShakeView : UIView {
}
@end

Сделайте ShakeView.m следующим образом:

#import "ShakeView.h"
@implementation ShakeView
- (BOOL)canBecomeFirstResponder {
    return YES;
}
- (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event {
    if ( event.subtype == UIEventSubtypeMotionShake ) {
        NSLog(@"Shake!");
    }

    if ([super respondsToSelector:@selector(motionEnded:withEvent:)]) {
        [super motionEnded:motion withEvent:event];
    }
}
@end

Сделайте ShakeTestViewController.h следующим образом:

#import <UIKit/UIKit.h>
#include "ShakeView.h"
@interface ShakeTestViewController : UIViewController {
    ShakeView *s;
}
@end

Сделайте ShakeTestViewController.m следующим образом:

#import "ShakeTestViewController.h"
@implementation ShakeTestViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    s = [[ShakeView alloc] init];
    [[self view] addSubview:s];
    [s becomeFirstResponder];
}
- (void)dealloc {
    [s release];
    [super dealloc];
}
@end

Создайте и запустите. Нажмите Cmd-Ctrl-Z, чтобы встряхнуть симулятор iPhone. Полюбуйтесь в сообщении журнала.

Ответ 2

У вас есть старый метод, реализованный

'acelerometer:didAccelerate'

? Если у вас реализован старый метод для всего приложения, вам нужно удалить его, прежде чем будет вызван новый.